2 * based on Windows Sockets 1.1 specs
3 * (ftp.microsoft.com:/Advsys/winsock/spec11/WINSOCK.TXT)
5 * (C) 1993,1994 John Brezak, Erik Bos.
11 #include <sys/types.h>
13 #include <sys/ioctl.h>
15 #include <sys/socket.h>
16 #include <netinet/in.h>
17 #include <arpa/inet.h>
27 static WORD wsa_errno;
28 static int wsa_initted;
29 static key_t wine_key = 0;
30 static FARPROC BlockFunction;
31 static fd_set fd_in_use;
40 #define IPC_PACKET_SIZE (sizeof(struct ipc_packet) - sizeof(long))
41 #define MTYPE 0xb0b0eb05
42 #define WINE_PACKED __attribute__ ((packed))
45 char *h_name WINE_PACKED; /* official name of host */
46 char **h_aliases WINE_PACKED; /* alias list */
47 int h_addrtype WINE_PACKED; /* host address type */
48 int h_length WINE_PACKED; /* length of address */
49 char **h_addr_list WINE_PACKED; /* list of addresses from name server */
55 char *p_name WINE_PACKED; /* official protocol name */
56 char **p_aliases WINE_PACKED; /* alias list */
57 int p_proto WINE_PACKED; /* protocol # */
61 char *s_name WINE_PACKED; /* official service name */
62 char **s_aliases WINE_PACKED; /* alias list */
63 int s_port WINE_PACKED; /* port # */
64 char *s_proto WINE_PACKED; /* protocol to use */
70 struct WIN_hostent hostent_addr;
71 struct WIN_hostent hostent_name;
72 struct WIN_protoent protoent_name;
73 struct WIN_protoent protoent_number;
74 struct WIN_servent servent_name;
75 struct WIN_servent servent_port;
77 struct WIN_hostent WSAhostent_addr;
78 struct WIN_hostent WSAhostent_name;
79 struct WIN_protoent WSAprotoent_name;
80 struct WIN_protoent WSAprotoent_number;
81 struct WIN_servent WSAservent_name;
82 struct WIN_servent WSAservent_port;
84 static struct WinSockHeap *heap;
86 #define dump_sockaddr(a) \
87 fprintf(stderr, "sockaddr_in: family %d, address %s, port %d\n", \
88 ((struct sockaddr_in *)a)->sin_family, \
89 inet_ntoa(((struct sockaddr_in *)a)->sin_addr), \
90 ntohs(((struct sockaddr_in *)a)->sin_port))
92 static WORD wsaerrno(void)
96 #if defined(__FreeBSD__)
97 fprintf(stderr, "winsock: errno %d, (%s).\n",
98 errno, sys_errlist[errno]);
100 fprintf(stderr, "winsock: errno %d, (%s).\n",
101 errno, strerror(errno));
104 fprintf(stderr, "winsock: errno %d\n", errno);
110 case EINTR: return WSAEINTR;
111 case EACCES: return WSAEACCES;
112 case EFAULT: return WSAEFAULT;
113 case EINVAL: return WSAEINVAL;
114 case EMFILE: return WSAEMFILE;
115 case EWOULDBLOCK: return WSAEWOULDBLOCK;
116 case EINPROGRESS: return WSAEINPROGRESS;
117 case EALREADY: return WSAEALREADY;
119 case ENOTSOCK: return WSAENOTSOCK;
120 case EDESTADDRREQ: return WSAEDESTADDRREQ;
121 case EMSGSIZE: return WSAEMSGSIZE;
122 case EPROTOTYPE: return WSAEPROTOTYPE;
123 case ENOPROTOOPT: return WSAENOPROTOOPT;
124 case EPROTONOSUPPORT: return WSAEPROTONOSUPPORT;
125 case ESOCKTNOSUPPORT: return WSAESOCKTNOSUPPORT;
126 case EOPNOTSUPP: return WSAEOPNOTSUPP;
127 case EPFNOSUPPORT: return WSAEPFNOSUPPORT;
128 case EAFNOSUPPORT: return WSAEAFNOSUPPORT;
129 case EADDRINUSE: return WSAEADDRINUSE;
130 case EADDRNOTAVAIL: return WSAEADDRNOTAVAIL;
131 case ENETDOWN: return WSAENETDOWN;
132 case ENETUNREACH: return WSAENETUNREACH;
133 case ENETRESET: return WSAENETRESET;
134 case ECONNABORTED: return WSAECONNABORTED;
135 case ECONNRESET: return WSAECONNRESET;
136 case ENOBUFS: return WSAENOBUFS;
137 case EISCONN: return WSAEISCONN;
138 case ENOTCONN: return WSAENOTCONN;
139 case ESHUTDOWN: return WSAESHUTDOWN;
140 case ETOOMANYREFS: return WSAETOOMANYREFS;
141 case ETIMEDOUT: return WSAETIMEDOUT;
142 case ECONNREFUSED: return WSAECONNREFUSED;
143 case ELOOP: return WSAELOOP;
144 case ENAMETOOLONG: return WSAENAMETOOLONG;
145 case EHOSTDOWN: return WSAEHOSTDOWN;
146 case EHOSTUNREACH: return WSAEHOSTUNREACH;
147 case ENOTEMPTY: return WSAENOTEMPTY;
149 case EPROCLIM: return WSAEPROCLIM;
151 case EUSERS: return WSAEUSERS;
152 case EDQUOT: return WSAEDQUOT;
153 case ESTALE: return WSAESTALE;
154 case EREMOTE: return WSAEREMOTE;
157 fprintf(stderr, "winsock: unknown errorno %d!\n", errno);
158 return WSAEOPNOTSUPP;
162 static void errno_to_wsaerrno(void)
164 wsa_errno = wsaerrno();
167 static void convert_sockopt(INT *level, INT *optname)
169 /* $%#%!@! why couldn't they use the same values for both winsock and unix ? */
175 case 0x01: *optname = SO_DEBUG;
177 case 0x04: *optname = SO_REUSEADDR;
179 case 0x08: *optname = SO_KEEPALIVE;
181 case 0x10: *optname = SO_DONTROUTE;
183 case 0x20: *optname = SO_BROADCAST;
185 case 0x80: *optname = SO_LINGER;
187 case 0x100: *optname = SO_OOBINLINE;
189 case 0x1001: *optname = SO_SNDBUF;
191 case 0x1002: *optname = SO_RCVBUF;
193 case 0x1007: *optname = SO_ERROR;
195 case 0x1008: *optname = SO_TYPE;
198 fprintf(stderr, "convert_sockopt() unknown optname %d\n", *optname);
202 case 6: *optname = IPPROTO_TCP;
207 static void CONVERT_HOSTENT(struct WIN_hostent *heap, struct hostent *host)
212 static void CONVERT_PROTOENT(struct WIN_protoent *heap, struct protoent *proto)
217 static void CONVERT_SERVENT(struct WIN_servent *heap, struct servent *serv)
222 #define CONVERT_HOSTENT(a,b) memcpy(a, &b, sizeof(a))
223 #define CONVERT_PROTOENT(a,b) memcpy(a, &b, sizeof(a))
224 #define CONVERT_SERVENT(a,b) memcpy(a, &b, sizeof(a))
227 SOCKET WINSOCK_accept(SOCKET s, struct sockaddr *addr, INT *addrlen)
231 dprintf_winsock(stddeb, "WSA_accept: socket %d, ptr %8x, length %d\n", s, (int) addr, *addrlen);
233 if ((sock = accept(s, addr, (int *) addrlen)) < 0) {
235 return INVALID_SOCKET;
240 INT WINSOCK_bind(SOCKET s, struct sockaddr *name, INT namelen)
242 dprintf_winsock(stddeb, "WSA_bind: socket %d, ptr %8x, length %d\n", s, (int) name, namelen);
245 if (bind(s, name, namelen) < 0) {
252 INT WINSOCK_closesocket(SOCKET s)
254 dprintf_winsock(stddeb, "WSA_closesocket: socket %d\n", s);
256 FD_CLR(s, &fd_in_use);
265 INT WINSOCK_connect(SOCKET s, struct sockaddr *name, INT namelen)
267 dprintf_winsock(stddeb, "WSA_connect: socket %d, ptr %8x, length %d\n", s, (int) name, namelen);
270 if (connect(s, name, namelen) < 0) {
277 INT WINSOCK_getpeername(SOCKET s, struct sockaddr *name, INT *namelen)
279 dprintf_winsock(stddeb, "WSA_getpeername: socket: %d, ptr %8x, ptr %8x\n", s, (int) name, *namelen);
282 if (getpeername(s, name, (int *) namelen) < 0) {
289 INT WINSOCK_getsockname(SOCKET s, struct sockaddr *name, INT *namelen)
291 dprintf_winsock(stddeb, "WSA_getsockname: socket: %d, ptr %8x, ptr %8x\n", s, (int) name, (int) *namelen);
292 if (getsockname(s, name, (int *) namelen) < 0) {
300 WINSOCK_getsockopt(SOCKET s, INT level, INT optname, char *optval, INT *optlen)
302 dprintf_winsock(stddeb, "WSA_getsockopt: socket: %d, opt %d, ptr %8x, ptr %8x\n", s, level, (int) optval, (int) *optlen);
303 convert_sockopt(&level, &optname);
305 if (getsockopt(s, (int) level, optname, optval, (int *) optlen) < 0) {
312 u_long WINSOCK_htonl(u_long hostlong)
314 return( htonl(hostlong) );
317 u_short WINSOCK_htons(u_short hostshort)
319 return( htons(hostshort) );
322 u_long WINSOCK_inet_addr(char *cp)
324 return( inet_addr(cp) );
327 char *WINSOCK_inet_ntoa(struct in_addr in)
331 /* dprintf_winsock(stddeb, "WSA_inet_ntoa: %8lx\n", (int) in);*/
333 if ((s = inet_ntoa(in)) == NULL) {
338 strncpy(heap->ntoa_buffer, s, sizeof(heap->ntoa_buffer) );
340 return (char *) &heap->ntoa_buffer;
343 INT WINSOCK_ioctlsocket(SOCKET s, long cmd, u_long *argp)
345 dprintf_winsock(stddeb, "WSA_ioctl: socket %d, cmd %ld, ptr %8x\n", s, cmd, (int) argp);
347 if (ioctl(s, cmd, argp) < 0) {
354 INT WINSOCK_listen(SOCKET s, INT backlog)
356 dprintf_winsock(stddeb, "WSA_listen: socket %d, backlog %d\n", s, backlog);
358 if (listen(s, backlog) < 0) {
365 u_long WINSOCK_ntohl(u_long netlong)
367 return( ntohl(netlong) );
370 u_short WINSOCK_ntohs(u_short netshort)
372 return( ntohs(netshort) );
375 INT WINSOCK_recv(SOCKET s, char *buf, INT len, INT flags)
379 dprintf_winsock(stddeb, "WSA_recv: socket %d, ptr %8x, length %d, flags %d\n", s, (int) buf, len, flags);
381 if ((length = recv(s, buf, len, flags)) < 0) {
388 INT WINSOCK_recvfrom(SOCKET s, char *buf, INT len, INT flags,
389 struct sockaddr *from, int *fromlen)
393 dprintf_winsock(stddeb, "WSA_recvfrom: socket %d, ptr %8lx, length %d, flags %d\n", s, (unsigned long)buf, len, flags);
395 if ((length = recvfrom(s, buf, len, flags, from, fromlen)) < 0) {
402 INT WINSOCK_select(INT nfds, fd_set *readfds, fd_set *writefds,
403 fd_set *exceptfds, struct timeval *timeout)
405 dprintf_winsock(stddeb, "WSA_select: fd # %d, ptr %8lx, ptr %8lx, ptr %8lX\n", nfds, (unsigned long) readfds, (unsigned long) writefds, (unsigned long) exceptfds);
407 return(select(nfds, readfds, writefds, exceptfds, timeout));
410 INT WINSOCK_send(SOCKET s, char *buf, INT len, INT flags)
414 dprintf_winsock(stddeb, "WSA_send: socket %d, ptr %8lx, length %d, flags %d\n", s, (unsigned long) buf, len, flags);
416 if ((length = send(s, buf, len, flags)) < 0) {
423 INT WINSOCK_sendto(SOCKET s, char *buf, INT len, INT flags,
424 struct sockaddr *to, INT tolen)
428 dprintf_winsock(stddeb, "WSA_sendto: socket %d, ptr %8lx, length %d, flags %d\n", s, (unsigned long) buf, len, flags);
430 if ((length = sendto(s, buf, len, flags, to, tolen)) < 0) {
437 INT WINSOCK_setsockopt(SOCKET s, INT level, INT optname, const char *optval,
440 dprintf_winsock(stddeb, "WSA_setsockopt: socket %d, level %d, opt %d, ptr %8x, len %d\n", s, level, optname, (int) optval, optlen);
441 convert_sockopt(&level, &optname);
443 if (setsockopt(s, level, optname, optval, optlen) < 0) {
450 INT WINSOCK_shutdown(SOCKET s, INT how)
452 dprintf_winsock(stddeb, "WSA_shutdown: socket s %d, how %d\n", s, how);
454 if (shutdown(s, how) < 0) {
461 SOCKET WINSOCK_socket(INT af, INT type, INT protocol)
465 dprintf_winsock(stddeb, "WSA_socket: af=%d type=%d protocol=%d\n", af, type, protocol);
467 if ((sock = socket(af, type, protocol)) < 0) {
469 dprintf_winsock(stddeb, "WSA_socket: failed !\n");
470 return INVALID_SOCKET;
474 wsa_errno = WSAEMFILE;
475 return INVALID_SOCKET;
478 FD_SET(sock, &fd_in_use);
480 dprintf_winsock(stddeb, "WSA_socket: fd %d\n", sock);
484 struct WIN_hostent *WINSOCK_gethostbyaddr(const char *addr, INT len, INT type)
486 struct hostent *host;
488 dprintf_winsock(stddeb, "WSA_gethostbyaddr: ptr %8x, len %d, type %d\n", (int) addr, len, type);
490 if ((host = gethostbyaddr(addr, len, type)) == NULL) {
494 CONVERT_HOSTENT(&heap->hostent_addr, host);
496 return &heap->hostent_addr;
499 struct WIN_hostent *WINSOCK_gethostbyname(const char *name)
501 struct hostent *host;
503 dprintf_winsock(stddeb, "WSA_gethostbyname: %s\n", name);
505 if ((host = gethostbyname(name)) == NULL) {
509 CONVERT_HOSTENT(&heap->hostent_name, host);
511 return &heap->hostent_name;
514 INT WINSOCK_gethostname(char *name, INT namelen)
516 dprintf_winsock(stddeb, "WSA_gethostname: name %s, len %d\n", name, namelen);
518 if (gethostname(name, namelen) < 0) {
525 struct WIN_protoent *WINSOCK_getprotobyname(char *name)
527 struct protoent *proto;
529 dprintf_winsock(stddeb, "WSA_getprotobyname: name %s\n", name);
531 if ((proto = getprotobyname(name)) == NULL) {
535 CONVERT_PROTOENT(&heap->protoent_name, proto);
537 return &heap->protoent_name;
540 struct WIN_protoent *WINSOCK_getprotobynumber(INT number)
542 struct protoent *proto;
544 dprintf_winsock(stddeb, "WSA_getprotobynumber: num %d\n", number);
546 if ((proto = getprotobynumber(number)) == NULL) {
550 CONVERT_PROTOENT(&heap->protoent_number, proto);
552 return &heap->protoent_number;
555 struct WIN_servent *WINSOCK_getservbyname(const char *name, const char *proto)
557 struct servent *service;
562 dprintf_winsock(stddeb, "WSA_getservbyname: name %s, proto %s\n", name, proto);
564 if ((service = getservbyname(name, proto)) == NULL) {
568 CONVERT_SERVENT(&heap->servent_name, service);
570 return &heap->servent_name;
573 struct WIN_servent *WINSOCK_getservbyport(INT port, const char *proto)
575 struct servent *service;
577 dprintf_winsock(stddeb, "WSA_getservbyport: port %d, name %s\n", port, proto);
579 if ((service = getservbyport(port, proto)) == NULL) {
583 CONVERT_SERVENT(&heap->servent_port, service);
585 return &heap->servent_port;
588 /******************** winsock specific functions ************************
591 static HANDLE new_handle = 0;
593 static HANDLE AllocWSAHandle(void)
598 static void recv_message(int sig)
600 struct ipc_packet message;
602 if (msgrcv(wine_key, &message, IPC_PACKET_SIZE, MTYPE, IPC_NOWAIT) == -1)
603 perror("wine: msgrcv");
606 "WSA: PostMessage (hwnd %d, wMsg %d, wParam %d, lParam %ld)\n",
612 PostMessage(message.hWnd, message.wMsg, message.handle, message.lParam);
614 signal(SIGUSR1, recv_message);
618 static void send_message(HANDLE handle, HWND hWnd, u_int wMsg, long lParam)
620 struct ipc_packet message;
622 message.mtype = MTYPE;
623 message.handle = handle;
626 message.lParam = lParam;
629 "WSA: send (hwnd %d, wMsg %d, handle %d, lParam %ld)\n",
630 hWnd, wMsg, handle, lParam);
632 if (msgsnd(wine_key, &message, IPC_PACKET_SIZE, IPC_NOWAIT) == -1)
633 perror("wine: msgsnd");
635 kill(getppid(), SIGUSR1);
639 HANDLE WSAAsyncGetHostByAddr(HWND hWnd, u_int wMsg, const char *addr,
640 INT len, INT type, char *buf, INT buflen)
643 struct hostent *host;
645 handle = AllocWSAHandle();
650 if ((host = gethostbyaddr(addr, len, type)) == NULL) {
651 send_message(hWnd, wMsg, handle, wsaerrno() << 16);
654 memcpy(buf, host, buflen);
655 send_message(hWnd, wMsg, handle, 0);
661 HANDLE WSAAsyncGetHostByName(HWND hWnd, u_int wMsg, const char *name,
662 char *buf, INT buflen)
665 struct hostent *host;
667 handle = AllocWSAHandle();
672 if ((host = gethostbyname(name)) == NULL) {
673 send_message(hWnd, wMsg, handle, wsaerrno() << 16);
676 memcpy(buf, host, buflen);
677 send_message(hWnd, wMsg, handle, 0);
683 HANDLE WSAAsyncGetProtoByName(HWND hWnd, u_int wMsg, const char *name,
684 char *buf, INT buflen)
687 struct protoent *proto;
689 handle = AllocWSAHandle();
694 if ((proto = getprotobyname(name)) == NULL) {
695 send_message(hWnd, wMsg, handle, wsaerrno() << 16);
698 memcpy(buf, proto, buflen);
699 send_message(hWnd, wMsg, handle, 0);
705 HANDLE WSAAsyncGetProtoByNumber(HWND hWnd, u_int wMsg, INT number,
706 char *buf, INT buflen)
709 struct protoent *proto;
711 handle = AllocWSAHandle();
716 if ((proto = getprotobynumber(number)) == NULL) {
717 send_message(hWnd, wMsg, handle, wsaerrno() << 16);
720 memcpy(buf, proto, buflen);
721 send_message(hWnd, wMsg, handle, 0);
727 HANDLE WSAAsyncGetServByName(HWND hWnd, u_int wMsg, const char *name,
728 const char *proto, char *buf, INT buflen)
731 struct servent *service;
733 handle = AllocWSAHandle();
738 if ((service = getservbyname(name, proto)) == NULL) {
739 send_message(hWnd, wMsg, handle, wsaerrno() << 16);
742 memcpy(buf, service, buflen);
743 send_message(hWnd, wMsg, handle, 0);
749 HANDLE WSAAsyncGetServByPort(HWND hWnd, u_int wMsg, INT port, const char
750 *proto, char *buf, INT buflen)
753 struct servent *service;
755 handle = AllocWSAHandle();
760 if ((service = getservbyport(port, proto)) == NULL) {
761 send_message(hWnd, wMsg, handle, wsaerrno() << 16);
764 memcpy(buf, service, buflen);
765 send_message(hWnd, wMsg, handle, 0);
770 INT WSAAsyncSelect(SOCKET s, HWND hWnd, u_int wMsg, long lEvent)
773 fd_set read_fds, write_fds, except_fds;
775 dprintf_winsock(stddeb, "WSA_AsyncSelect: socket %d, HWND %d, wMsg %d, event %ld\n", s, hWnd, wMsg, lEvent);
777 /* remove outstanding asyncselect() processes */
780 if (wMsg == 0 && lEvent == 0)
789 FD_ZERO(&except_fds);
791 if (lEvent & FD_READ)
792 FD_SET(s, &read_fds);
793 if (lEvent & FD_WRITE)
794 FD_SET(s, &write_fds);
796 fcntl(s, F_SETFL, O_NONBLOCK);
797 select(s + 1, &read_fds, &write_fds, &except_fds, NULL);
800 if (FD_ISSET(s, &read_fds))
802 if (FD_ISSET(s, &write_fds))
805 send_message(hWnd, wMsg, s, (wsaerrno() << 16) | event);
810 INT WSAFDIsSet(INT fd, fd_set *set)
812 return( FD_ISSET(fd, set) );
815 INT WSACancelAsyncRequest(HANDLE hAsyncTaskHandle)
817 dprintf_winsock(stddeb, "WSA_AsyncRequest: handle %d\n", hAsyncTaskHandle);
822 INT WSACancelBlockingCall(void)
824 dprintf_winsock(stddeb, "WSA_CancelBlockCall\n");
828 INT WSAGetLastError(void)
830 dprintf_winsock(stddeb, "WSA_GetLastError\n");
835 void WSASetLastError(INT iError)
837 dprintf_winsock(stddeb, "WSA_SetLastErorr %d\n", iError);
842 BOOL WSAIsBlocking(void)
844 dprintf_winsock(stddeb, "WSA_IsBlocking\n");
849 FARPROC WSASetBlockingHook(FARPROC lpBlockFunc)
851 dprintf_winsock(stddeb, "WSA_SetBlockHook %8lx, STUB!\n", (unsigned long) lpBlockFunc);
852 BlockFunction = lpBlockFunc;
854 return (FARPROC) lpBlockFunc;
857 INT WSAUnhookBlockingHook(void)
859 dprintf_winsock(stddeb, "WSA_UnhookBlockingHook\n");
860 BlockFunction = NULL;
865 WSADATA WINSOCK_data = {
883 INT WSAStartup(WORD wVersionRequested, LPWSADATA lpWSAData)
888 dprintf_winsock(stddeb, "WSAStartup: verReq=%x\n", wVersionRequested);
890 if (LOBYTE(wVersionRequested) < 1 ||
891 (LOBYTE(wVersionRequested) == 1 &&
892 HIBYTE(wVersionRequested) < 1))
893 return WSAVERNOTSUPPORTED;
898 /* alloc winsock heap */
900 if ((HeapHandle = GlobalAlloc(GMEM_FIXED,sizeof(struct WinSockHeap))) == 0)
901 return WSASYSNOTREADY;
903 heap = (struct WinSockHeap *) GlobalLock(HeapHandle);
905 HEAP_Init(&MyHeap, heap, sizeof(struct WinSockHeap));
907 bcopy(&WINSOCK_data, lpWSAData, sizeof(WINSOCK_data));
911 if ((wine_key = msgget(IPC_PRIVATE, 0600)) == -1)
912 perror("wine: msgget");
914 signal(SIGUSR1, recv_message);
929 if (msgctl(wine_key, IPC_RMID, NULL) == -1)
930 perror("wine: shmctl");
932 for (fd = 0; fd != FD_SETSIZE; fd++)
933 if (FD_ISSET(fd, &fd_in_use))