| 1 | /* $Id: wsa.cpp,v 1.6 2000-05-02 13:09:43 bird Exp $ */ | 
|---|
| 2 |  | 
|---|
| 3 | /* | 
|---|
| 4 | * | 
|---|
| 5 | * Win32 SOCK32 for OS/2 (WSA apis) | 
|---|
| 6 | * | 
|---|
| 7 | * Copyright (C) 2000 Sander van Leeuwen (sandervl@xs4all.nl) | 
|---|
| 8 | * | 
|---|
| 9 | * Based on Wine code: (dlls\winsock\async.c,socket.c) | 
|---|
| 10 | * (C) 1993,1994,1996,1997 John Brezak, Erik Bos, Alex Korobka. | 
|---|
| 11 | * (C) 1999 Marcus Meissner | 
|---|
| 12 | * | 
|---|
| 13 | * Project Odin Software License can be found in LICENSE.TXT | 
|---|
| 14 | * | 
|---|
| 15 | */ | 
|---|
| 16 | #define INCL_BASE | 
|---|
| 17 | #include <os2wrap.h> | 
|---|
| 18 | #include <stdio.h> | 
|---|
| 19 | #include <string.h> | 
|---|
| 20 | #include <odin.h> | 
|---|
| 21 | #include <odinwrap.h> | 
|---|
| 22 | #include <os2sel.h> | 
|---|
| 23 | #include <misc.h> | 
|---|
| 24 | #include <win32api.h> | 
|---|
| 25 |  | 
|---|
| 26 | #include "wsock32.h" | 
|---|
| 27 | #include "asyncthread.h" | 
|---|
| 28 |  | 
|---|
| 29 | #define DBG_LOCALLOG    DBG_wsa | 
|---|
| 30 | #include "dbglocal.h" | 
|---|
| 31 |  | 
|---|
| 32 | ODINDEBUGCHANNEL(WSOCK32-WSA) | 
|---|
| 33 |  | 
|---|
| 34 | //kso: dirty fix to make this compile! not permanent! | 
|---|
| 35 | BOOL WINAPI QueryPerformanceCounter(LARGE_INTEGER *p); | 
|---|
| 36 | #define LowPart u.LowPart | 
|---|
| 37 |  | 
|---|
| 38 |  | 
|---|
| 39 | #ifndef winsockcleanupsockets | 
|---|
| 40 | // Exported by SO32DLL.  Used to close all open sockets. | 
|---|
| 41 | void _System winsockcleanupsockets(void); | 
|---|
| 42 | #endif | 
|---|
| 43 |  | 
|---|
| 44 | static void WSASetBlocking(BOOL fBlock); | 
|---|
| 45 |  | 
|---|
| 46 | BOOL fWSAInitialized = FALSE; | 
|---|
| 47 | static int num_startup = 0; | 
|---|
| 48 |  | 
|---|
| 49 | //****************************************************************************** | 
|---|
| 50 | WSADATA WINSOCK_data = { 0x0101, 0x0101, | 
|---|
| 51 | "Odin Sockets 1.1", "OS/2", | 
|---|
| 52 | WS_MAX_SOCKETS_PER_PROCESS, | 
|---|
| 53 | WS_MAX_UDP_DATAGRAM, NULL }; | 
|---|
| 54 | //****************************************************************************** | 
|---|
| 55 | ODINFUNCTION2(int, WSAStartup, | 
|---|
| 56 | USHORT,wVersionRequested, | 
|---|
| 57 | LPWSADATA,lpWSAData) | 
|---|
| 58 | { | 
|---|
| 59 | LPWSINFO            pwsi; | 
|---|
| 60 |  | 
|---|
| 61 | if (LOBYTE(wVersionRequested) < 1 || (LOBYTE(wVersionRequested) == 1 && | 
|---|
| 62 | HIBYTE(wVersionRequested) < 1)) return WSAVERNOTSUPPORTED; | 
|---|
| 63 |  | 
|---|
| 64 | if (!lpWSAData) return WSAEINVAL; | 
|---|
| 65 |  | 
|---|
| 66 | if(fWSAInitialized == FALSE) | 
|---|
| 67 | { | 
|---|
| 68 | WINSOCK_CreateIData(); | 
|---|
| 69 | pwsi = WINSOCK_GetIData(); | 
|---|
| 70 |  | 
|---|
| 71 | sock_init(); | 
|---|
| 72 | } | 
|---|
| 73 | else pwsi = WINSOCK_GetIData(); | 
|---|
| 74 |  | 
|---|
| 75 | if (!pwsi) return WSASYSNOTREADY; | 
|---|
| 76 |  | 
|---|
| 77 | num_startup++; | 
|---|
| 78 | fWSAInitialized = TRUE; | 
|---|
| 79 |  | 
|---|
| 80 | /* return winsock information */ | 
|---|
| 81 | memcpy(lpWSAData, &WINSOCK_data, sizeof(WINSOCK_data)); | 
|---|
| 82 |  | 
|---|
| 83 | return 0; | 
|---|
| 84 | } | 
|---|
| 85 | //****************************************************************************** | 
|---|
| 86 | //****************************************************************************** | 
|---|
| 87 | ODINFUNCTION0(int,WSACleanup) | 
|---|
| 88 | { | 
|---|
| 89 | LPWSINFO pwsi = WINSOCK_GetIData(); | 
|---|
| 90 | if( pwsi ) { | 
|---|
| 91 | if( --num_startup > 0 ) return 0; | 
|---|
| 92 |  | 
|---|
| 93 | winsockcleanupsockets(); | 
|---|
| 94 |  | 
|---|
| 95 | fWSAInitialized = FALSE; | 
|---|
| 96 | WINSOCK_DeleteIData(); | 
|---|
| 97 | return 0; | 
|---|
| 98 | } | 
|---|
| 99 | SetLastError(WSANOTINITIALISED); | 
|---|
| 100 | return SOCKET_ERROR; | 
|---|
| 101 | } | 
|---|
| 102 | //****************************************************************************** | 
|---|
| 103 | //TODO: Implement this | 
|---|
| 104 | //****************************************************************************** | 
|---|
| 105 | ODINFUNCTION1(FARPROC,WSASetBlockingHook, | 
|---|
| 106 | FARPROC,lpBlockFunc) | 
|---|
| 107 | { | 
|---|
| 108 | FARPROC             prev; | 
|---|
| 109 | LPWSINFO            pwsi = WINSOCK_GetIData(); | 
|---|
| 110 |  | 
|---|
| 111 | if( pwsi ) { | 
|---|
| 112 | prev = (FARPROC)pwsi->blocking_hook; | 
|---|
| 113 | pwsi->blocking_hook = (DWORD)lpBlockFunc; | 
|---|
| 114 | pwsi->flags |= WSI_BLOCKINGHOOK; | 
|---|
| 115 | return prev; | 
|---|
| 116 | } | 
|---|
| 117 | return NULL; | 
|---|
| 118 | } | 
|---|
| 119 | //****************************************************************************** | 
|---|
| 120 | //TODO: Implement this | 
|---|
| 121 | //****************************************************************************** | 
|---|
| 122 | ODINFUNCTION0(int,WSAUnhookBlockingHook) | 
|---|
| 123 | { | 
|---|
| 124 | LPWSINFO pwsi = WINSOCK_GetIData(); | 
|---|
| 125 |  | 
|---|
| 126 | if( pwsi ) | 
|---|
| 127 | { | 
|---|
| 128 | pwsi->blocking_hook = 0; | 
|---|
| 129 | pwsi->flags &= ~WSI_BLOCKINGHOOK; | 
|---|
| 130 | return 0; | 
|---|
| 131 | } | 
|---|
| 132 | return SOCKET_ERROR; | 
|---|
| 133 | } | 
|---|
| 134 | //****************************************************************************** | 
|---|
| 135 | //****************************************************************************** | 
|---|
| 136 | BOOL WIN32API WSAIsBlocking() | 
|---|
| 137 | { | 
|---|
| 138 | LPWSINFO pwsi = WINSOCK_GetIData(); | 
|---|
| 139 |  | 
|---|
| 140 | if( pwsi ) | 
|---|
| 141 | { | 
|---|
| 142 | return (pwsi->flags & WSI_BLOCKINGCALL) == WSI_BLOCKINGCALL; | 
|---|
| 143 | } | 
|---|
| 144 | DebugInt3(); | 
|---|
| 145 | return FALSE; //shouldn't happen | 
|---|
| 146 | } | 
|---|
| 147 | //****************************************************************************** | 
|---|
| 148 | //****************************************************************************** | 
|---|
| 149 | void WSASetBlocking(BOOL fBlock, HANDLE tid) | 
|---|
| 150 | { | 
|---|
| 151 | LPWSINFO pwsi = WINSOCK_GetIData(tid); | 
|---|
| 152 |  | 
|---|
| 153 | if( pwsi ) | 
|---|
| 154 | { | 
|---|
| 155 | if(fBlock) { | 
|---|
| 156 | pwsi->flags |= WSI_BLOCKINGCALL; | 
|---|
| 157 | } | 
|---|
| 158 | else    pwsi->flags &= ~WSI_BLOCKINGCALL; | 
|---|
| 159 | return; | 
|---|
| 160 | } | 
|---|
| 161 | DebugInt3(); | 
|---|
| 162 | } | 
|---|
| 163 | //****************************************************************************** | 
|---|
| 164 | //****************************************************************************** | 
|---|
| 165 | ODINFUNCTION2(int,__WSAFDIsSet,SOCKET, s, ws_fd_set*,set) | 
|---|
| 166 | { | 
|---|
| 167 | int i = set->fd_count; | 
|---|
| 168 |  | 
|---|
| 169 | while (i--) | 
|---|
| 170 | if (set->fd_array[i] == s) return 1; | 
|---|
| 171 |  | 
|---|
| 172 | return 0; | 
|---|
| 173 | } | 
|---|
| 174 | //****************************************************************************** | 
|---|
| 175 | //****************************************************************************** | 
|---|
| 176 | UINT wsaErrno() | 
|---|
| 177 | { | 
|---|
| 178 | int loc_errno = sock_errno(); | 
|---|
| 179 |  | 
|---|
| 180 | //Warp 4 returns errors with base 0 | 
|---|
| 181 | if(loc_errno && loc_errno < SOCBASEERR) { | 
|---|
| 182 | loc_errno += SOCBASEERR; | 
|---|
| 183 | } | 
|---|
| 184 | switch(loc_errno) | 
|---|
| 185 | { | 
|---|
| 186 | case SOCEINTR:          return WSAEINTR; | 
|---|
| 187 | case SOCEBADF:          return WSAEBADF; | 
|---|
| 188 | case SOCEPERM: | 
|---|
| 189 | case SOCEACCES:         return WSAEACCES; | 
|---|
| 190 | case SOCEFAULT:         return WSAEFAULT; | 
|---|
| 191 | case SOCEINVAL:         return WSAEINVAL; | 
|---|
| 192 | case SOCEMFILE:         return WSAEMFILE; | 
|---|
| 193 | case SOCEWOULDBLOCK:    return WSAEWOULDBLOCK; | 
|---|
| 194 | case SOCEINPROGRESS:    return WSAEINPROGRESS; | 
|---|
| 195 | case SOCEALREADY:       return WSAEALREADY; | 
|---|
| 196 | case SOCENOTSOCK:       return WSAENOTSOCK; | 
|---|
| 197 | case SOCEDESTADDRREQ:   return WSAEDESTADDRREQ; | 
|---|
| 198 | case SOCEMSGSIZE:       return WSAEMSGSIZE; | 
|---|
| 199 | case SOCEPROTOTYPE:     return WSAEPROTOTYPE; | 
|---|
| 200 | case SOCENOPROTOOPT:    return WSAENOPROTOOPT; | 
|---|
| 201 | case SOCEPROTONOSUPPORT:return WSAEPROTONOSUPPORT; | 
|---|
| 202 | case SOCESOCKTNOSUPPORT:return WSAESOCKTNOSUPPORT; | 
|---|
| 203 | case SOCEOPNOTSUPP:     return WSAEOPNOTSUPP; | 
|---|
| 204 | case SOCEPFNOSUPPORT:   return WSAEPFNOSUPPORT; | 
|---|
| 205 | case SOCEAFNOSUPPORT:   return WSAEAFNOSUPPORT; | 
|---|
| 206 | case SOCEADDRINUSE:     return WSAEADDRINUSE; | 
|---|
| 207 | case SOCEADDRNOTAVAIL:  return WSAEADDRNOTAVAIL; | 
|---|
| 208 | case SOCENETDOWN:       return WSAENETDOWN; | 
|---|
| 209 | case SOCENETUNREACH:    return WSAENETUNREACH; | 
|---|
| 210 | case SOCENETRESET:      return WSAENETRESET; | 
|---|
| 211 | case SOCECONNABORTED:   return WSAECONNABORTED; | 
|---|
| 212 | case SOCEPIPE: | 
|---|
| 213 | case SOCECONNRESET:     return WSAECONNRESET; | 
|---|
| 214 | case SOCENOBUFS:        return WSAENOBUFS; | 
|---|
| 215 | case SOCEISCONN:        return WSAEISCONN; | 
|---|
| 216 | case SOCENOTCONN:       return WSAENOTCONN; | 
|---|
| 217 | case SOCESHUTDOWN:      return WSAESHUTDOWN; | 
|---|
| 218 | case SOCETOOMANYREFS:   return WSAETOOMANYREFS; | 
|---|
| 219 | case SOCETIMEDOUT:      return WSAETIMEDOUT; | 
|---|
| 220 | case SOCECONNREFUSED:   return WSAECONNREFUSED; | 
|---|
| 221 | case SOCELOOP:          return WSAELOOP; | 
|---|
| 222 | case SOCENAMETOOLONG:   return WSAENAMETOOLONG; | 
|---|
| 223 | case SOCEHOSTDOWN:      return WSAEHOSTDOWN; | 
|---|
| 224 | case SOCEHOSTUNREACH:   return WSAEHOSTUNREACH; | 
|---|
| 225 | case SOCENOTEMPTY:      return WSAENOTEMPTY; | 
|---|
| 226 | case SOCENXIO: | 
|---|
| 227 | case SOCESRCH: | 
|---|
| 228 | case SOCEOS2ERR:        return WSAEFAULT; | 
|---|
| 229 |  | 
|---|
| 230 | /* just in case we ever get here and there are no problems */ | 
|---|
| 231 | case 0:                 return 0; | 
|---|
| 232 | default: | 
|---|
| 233 | dprintf(("Unknown errno %d!\n", loc_errno)); | 
|---|
| 234 | return WSAEOPNOTSUPP; | 
|---|
| 235 | } | 
|---|
| 236 | } | 
|---|
| 237 | //****************************************************************************** | 
|---|
| 238 | //****************************************************************************** | 
|---|
| 239 | UINT wsaHerrno() | 
|---|
| 240 | { | 
|---|
| 241 | int loc_errno = h_errno; | 
|---|
| 242 |  | 
|---|
| 243 | switch(loc_errno) | 
|---|
| 244 | { | 
|---|
| 245 | case HOST_NOT_FOUND:    return WSAHOST_NOT_FOUND; | 
|---|
| 246 | case TRY_AGAIN:         return WSATRY_AGAIN; | 
|---|
| 247 | case NO_RECOVERY:       return WSANO_RECOVERY; | 
|---|
| 248 | case NO_DATA:           return WSANO_DATA; | 
|---|
| 249 |  | 
|---|
| 250 | case 0:                 return 0; | 
|---|
| 251 | default: | 
|---|
| 252 | dprintf(("Unknown h_errno %d!\n", loc_errno)); | 
|---|
| 253 | return WSAEOPNOTSUPP; | 
|---|
| 254 | } | 
|---|
| 255 | } | 
|---|
| 256 | //****************************************************************************** | 
|---|
| 257 | //****************************************************************************** | 
|---|