Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2290 lines
49 KiB

  1. /*++
  2. Copyright (c) 1992 Microsoft Corporation
  3. Module Name:
  4. Wsraw.h
  5. Abstract:
  6. Support for raw winsock calls for WOW.
  7. Author:
  8. David Treadwell (davidtr) 02-Oct-1992
  9. Revision History:
  10. --*/
  11. #include "precomp.h"
  12. #pragma hdrstop
  13. #include "wsdynmc.h"
  14. LIST_ENTRY WWS32SocketHandleListHead;
  15. WORD WWS32SocketHandleCounter;
  16. BOOL WWS32SocketHandleCounterWrapped;
  17. //
  18. // The (PCHAR) casts in the following macro force the compiler to assume
  19. // only BYTE alignment.
  20. //
  21. #define SockCopyMemory(d,s,l) RtlCopyMemory( (PCHAR)(d), (PCHAR)(s), (l) )
  22. #define WSEXIT_IF_NOT_INTIALIZED() \
  23. if(!WWS32IsThreadInitialized) { \
  24. SetLastError(WSANOTINITIALISED); \
  25. RETURN((ULONG)SOCKET_ERROR); \
  26. }
  27. int SocketOption16To32(IN WORD SocketOption16);
  28. DWORD WSGetWinsock32(IN HAND16 h16,
  29. OUT PULONG pul);
  30. BOOL WSThunkAddrBufAndLen(IN PSOCKADDR fastSockaddr,
  31. IN VPSOCKADDR vpSockAddr16,
  32. IN VPWORD vpwAddrLen16,
  33. OUT PINT addressLength,
  34. OUT PINT *pAddressLength,
  35. OUT PSOCKADDR *realSockaddr);
  36. VOID WSUnThunkAddrBufAndLen(IN ULONG ret,
  37. IN VPWORD vpwAddrLen16,
  38. IN VPSOCKADDR vpSockAddr16,
  39. IN INT addressLength,
  40. IN PSOCKADDR fastSockaddr,
  41. IN PSOCKADDR realSockaddr);
  42. BOOL WSThunkAddrBuf(IN INT addressLength,
  43. IN VPSOCKADDR vpSockAddr16,
  44. IN PSOCKADDR fastSockaddr,
  45. OUT PSOCKADDR *realSockaddr);
  46. VOID WSUnThunkAddrBuf(IN PSOCKADDR fastSockaddr,
  47. IN PSOCKADDR realSockaddr);
  48. BOOL WSThunkRecvBuffer(IN INT BufferLength,
  49. IN VPBYTE vpBuf16,
  50. OUT PBYTE *buffer);
  51. VOID WSUnthunkRecvBuffer(IN INT cBytes,
  52. IN INT BufferLength,
  53. IN VPBYTE vpBuf16,
  54. IN PBYTE buffer);
  55. BOOL WSThunkSendBuffer(IN INT BufferLength,
  56. IN VPBYTE vpBuf16,
  57. OUT PBYTE *buffer);
  58. VOID WSUnthunkSendBuffer(IN PBYTE buffer);
  59. /*++
  60. GENERIC FUNCTION PROTOTYPE:
  61. ==========================
  62. ULONG FASTCALL WWS32<function name>(PVDMFRAME pFrame)
  63. {
  64. ULONG ul;
  65. register P<function name>16 parg16;
  66. GETARGPTR(pFrame, sizeof(<function name>16), parg16);
  67. <get any other required pointers into 16 bit space>
  68. ALLOCVDMPTR
  69. GETVDMPTR
  70. GETMISCPTR
  71. et cetera
  72. <copy any complex structures from 16 bit -> 32 bit space>
  73. <ALWAYS use the FETCHxxx macros>
  74. ul = GET<return type>16(<function name>(parg16->f1,
  75. :
  76. :
  77. parg16->f<n>);
  78. <copy any complex structures from 32 -> 16 bit space>
  79. <ALWAYS use the STORExxx macros>
  80. <free any pointers to 16 bit space you previously got>
  81. <flush any areas of 16 bit memory if they were written to>
  82. FLUSHVDMPTR
  83. FREEARGPTR( parg16 );
  84. RETURN( ul );
  85. }
  86. NOTE:
  87. The VDM frame is automatically set up, with all the function parameters
  88. available via parg16->f<number>.
  89. Handles must ALWAYS be mapped for 16 -> 32 -> 16 space via the mapping tables
  90. laid out in WALIAS.C.
  91. Any storage you allocate must be freed (eventually...).
  92. Further to that - if a thunk which allocates memory fails in the 32 bit call
  93. then it must free that memory.
  94. Also, never update structures in 16 bit land if the 32 bit call fails.
  95. Be aware that the GETxxxPTR macros return the CURRENT selector-to-flat_memory
  96. mapping. Calls to some 32-bit functions may indirectly cause callbacks into
  97. 16-bit code. These may cause 16-bit memory to move due to allocations
  98. made in 16-bit land. If the 16-bit memory does move, the corresponding 32-bit
  99. ptr in WOW32 needs to be refreshed to reflect the NEW selector-to-flat_memory
  100. mapping.
  101. --*/
  102. ULONG FASTCALL WWS32accept(PVDMFRAME pFrame)
  103. {
  104. ULONG ul = GETWORD16(INVALID_SOCKET);
  105. register PACCEPT16 parg16;
  106. SOCKET s32;
  107. SOCKET news32;
  108. HSOCKET16 news16;
  109. INT addressLength;
  110. PINT pAddressLength;
  111. SOCKADDR fastSockaddr;
  112. PSOCKADDR realSockaddr;
  113. VPWORD vpwAddrLen16;
  114. VPSOCKADDR vpSockAddr16;
  115. WSEXIT_IF_NOT_INTIALIZED();
  116. GETARGPTR(pFrame, sizeof(PACCEPT16), parg16);
  117. //
  118. // Find the 32-bit socket handle.
  119. //
  120. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  121. goto exit;
  122. }
  123. vpwAddrLen16 = (VPWORD)FETCHDWORD(parg16->AddressLength);
  124. vpSockAddr16 = (VPSOCKADDR)FETCHDWORD(parg16->Address);
  125. // Thunk the 16-bit Address name and length buffers
  126. if(!WSThunkAddrBufAndLen(&fastSockaddr,
  127. vpSockAddr16,
  128. vpwAddrLen16,
  129. &addressLength,
  130. &pAddressLength,
  131. &realSockaddr)) {
  132. goto exit;
  133. }
  134. // call the 32-bit API
  135. news32 = (*wsockapis[WOW_ACCEPT].lpfn)( s32, realSockaddr, pAddressLength);
  136. // Note: 16-bit callbacks resulting from above function
  137. // call may have caused 16-bit memory movement
  138. FREEARGPTR(pFrame);
  139. FREEARGPTR(parg16);
  140. // Un-Thunk the 16-bit Address name and length buffers
  141. WSUnThunkAddrBufAndLen((ULONG)news32,
  142. vpwAddrLen16,
  143. vpSockAddr16,
  144. addressLength,
  145. &fastSockaddr,
  146. realSockaddr);
  147. //
  148. // If the call succeeded, alias the 32-bit socket handle we just
  149. // obtained into a 16-bit handle.
  150. //
  151. if ( news32 != INVALID_SOCKET ) {
  152. news16 = GetWinsock16( news32, 0 );
  153. if ( news16 == 0 ) {
  154. (*wsockapis[WOW_CLOSESOCKET].lpfn)( news32 );
  155. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAENOBUFS );
  156. // Note: 16-bit callbacks resulting from above function
  157. // call may have caused 16-bit memory movement
  158. goto exit;
  159. }
  160. ul = news16;
  161. }
  162. exit:
  163. FREEARGPTR( parg16 );
  164. RETURN( ul );
  165. } // WWS32accept
  166. ULONG FASTCALL WWS32bind(PVDMFRAME pFrame)
  167. {
  168. ULONG ul = GETWORD16(INVALID_SOCKET);
  169. register PBIND16 parg16;
  170. SOCKET s32;
  171. SOCKADDR fastSockaddr;
  172. PSOCKADDR realSockaddr;
  173. INT addressLength;
  174. VPSOCKADDR vpSockAddr16;
  175. WSEXIT_IF_NOT_INTIALIZED();
  176. GETARGPTR(pFrame, sizeof(PBIND16), parg16);
  177. //
  178. // Find the 32-bit socket handle.
  179. //
  180. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  181. goto exit;
  182. }
  183. vpSockAddr16 = (VPSOCKADDR)FETCHDWORD(parg16->Address);
  184. addressLength = INT32(parg16->AddressLength);
  185. // Thunk the 16-bit address buffer
  186. if(!WSThunkAddrBuf(addressLength,
  187. vpSockAddr16,
  188. &fastSockaddr,
  189. &realSockaddr)) {
  190. goto exit;
  191. }
  192. ul = GETWORD16( (*wsockapis[WOW_BIND].lpfn)(s32,
  193. realSockaddr,
  194. addressLength));
  195. // Note: 16-bit callbacks resulting from above function
  196. // call may have caused 16-bit memory movement
  197. FREEARGPTR(pFrame);
  198. FREEARGPTR(parg16);
  199. // Un-Thunk the 16-bit address buffer
  200. WSUnThunkAddrBuf(&fastSockaddr, realSockaddr);
  201. exit:
  202. FREEARGPTR( parg16 );
  203. RETURN( ul );
  204. } // WWS32bind
  205. ULONG FASTCALL WWS32closesocket(PVDMFRAME pFrame)
  206. {
  207. ULONG ul = GETWORD16(INVALID_SOCKET);
  208. register PCLOSESOCKET16 parg16;
  209. SOCKET s32;
  210. HSOCKET16 hSocket16;
  211. WSEXIT_IF_NOT_INTIALIZED();
  212. GETARGPTR(pFrame, sizeof(CLOSESOCKET16), parg16);
  213. hSocket16 = (HSOCKET16)FETCHWORD(parg16->hSocket);
  214. //
  215. // Find the 32-bit socket handle.
  216. //
  217. if((s32 = WSGetWinsock32(hSocket16, &ul)) == INVALID_SOCKET) {
  218. goto exit;
  219. }
  220. ul = GETWORD16( (*wsockapis[WOW_CLOSESOCKET].lpfn)( s32 ) );
  221. // Note: 16-bit callbacks resulting from above function
  222. // call may have caused 16-bit memory movement
  223. FREEARGPTR(pFrame);
  224. FREEARGPTR(parg16);
  225. exit:
  226. //
  227. // Free the space in the alias table.
  228. //
  229. FreeWinsock16( hSocket16 );
  230. FREEARGPTR( parg16 );
  231. RETURN( ul );
  232. } // WWS32closesocket
  233. ULONG FASTCALL WWS32connect(PVDMFRAME pFrame)
  234. {
  235. ULONG ul = GETWORD16(INVALID_SOCKET);
  236. register PCONNECT16 parg16;
  237. SOCKET s32;
  238. SOCKADDR fastSockaddr;
  239. PSOCKADDR realSockaddr;
  240. INT addressLength;
  241. VPSOCKADDR vpSockAddr16;
  242. WSEXIT_IF_NOT_INTIALIZED();
  243. GETARGPTR(pFrame, sizeof(PCONNECT16), parg16);
  244. vpSockAddr16 = (VPSOCKADDR)FETCHDWORD(parg16->Address);
  245. addressLength = INT32(parg16->AddressLength);
  246. //
  247. // Find the 32-bit socket handle.
  248. //
  249. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  250. goto exit;
  251. }
  252. // Thunk the 16-bit address buffer
  253. if(!WSThunkAddrBuf(addressLength,
  254. vpSockAddr16,
  255. &fastSockaddr,
  256. &realSockaddr)) {
  257. goto exit;
  258. }
  259. ul = GETWORD16( (*wsockapis[WOW_CONNECT].lpfn)(s32,
  260. realSockaddr,
  261. addressLength));
  262. // Note: 16-bit callbacks resulting from above function
  263. // call may have caused 16-bit memory movement
  264. FREEARGPTR(pFrame);
  265. FREEARGPTR(parg16);
  266. // Un-Thunk the 16-bit address buffer
  267. WSUnThunkAddrBuf(&fastSockaddr, realSockaddr);
  268. exit:
  269. FREEARGPTR( parg16 );
  270. RETURN( ul );
  271. } // WWS32connect
  272. ULONG FASTCALL WWS32getpeername(PVDMFRAME pFrame)
  273. {
  274. ULONG ul = GETWORD16(INVALID_SOCKET);
  275. register PGETPEERNAME16 parg16;
  276. SOCKET s32;
  277. INT addressLength;
  278. PINT pAddressLength;
  279. SOCKADDR fastSockaddr;
  280. PSOCKADDR realSockaddr;
  281. VPWORD vpwAddrLen16;
  282. VPSOCKADDR vpSockAddr16;
  283. WSEXIT_IF_NOT_INTIALIZED();
  284. GETARGPTR(pFrame, sizeof(PGETPEERNAME16), parg16);
  285. //
  286. // Find the 32-bit socket handle.
  287. //
  288. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  289. goto exit;
  290. }
  291. vpSockAddr16 = (VPSOCKADDR)FETCHDWORD(parg16->Address);
  292. vpwAddrLen16 = (VPWORD)FETCHDWORD(parg16->AddressLength);
  293. // Thunk the 16-bit Address name and length buffers
  294. if(!WSThunkAddrBufAndLen(&fastSockaddr,
  295. vpSockAddr16,
  296. vpwAddrLen16,
  297. &addressLength,
  298. &pAddressLength,
  299. &realSockaddr)) {
  300. goto exit;
  301. }
  302. ul = GETWORD16( (*wsockapis[WOW_GETPEERNAME].lpfn)(s32,
  303. realSockaddr,
  304. pAddressLength));
  305. // Note: 16-bit callbacks resulting from above function
  306. // call may have caused 16-bit memory movement
  307. FREEARGPTR(pFrame);
  308. FREEARGPTR(parg16);
  309. // Un-Thunk the 16-bit Address name and length buffers
  310. WSUnThunkAddrBufAndLen(ul,
  311. vpwAddrLen16,
  312. vpSockAddr16,
  313. addressLength,
  314. &fastSockaddr,
  315. realSockaddr);
  316. exit:
  317. FREEARGPTR( parg16 );
  318. RETURN( ul );
  319. } // WWS32getpeername
  320. ULONG FASTCALL WWS32getsockname(PVDMFRAME pFrame)
  321. {
  322. ULONG ul = GETWORD16(INVALID_SOCKET);
  323. register PGETSOCKNAME16 parg16;
  324. SOCKET s32;
  325. INT addressLength;
  326. PINT pAddressLength;
  327. SOCKADDR fastSockaddr;
  328. PSOCKADDR realSockaddr;
  329. VPWORD vpwAddrLen16;
  330. VPSOCKADDR vpSockAddr16;
  331. WSEXIT_IF_NOT_INTIALIZED();
  332. GETARGPTR(pFrame, sizeof(PGETSOCKNAME16), parg16);
  333. //
  334. // Find the 32-bit socket handle.
  335. //
  336. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  337. goto exit;
  338. }
  339. vpSockAddr16 = (VPSOCKADDR)FETCHDWORD(parg16->Address);
  340. vpwAddrLen16 = (VPWORD)FETCHDWORD(parg16->AddressLength);
  341. // Thunk the 16-bit Address name and length buffers
  342. if(!WSThunkAddrBufAndLen(&fastSockaddr,
  343. vpSockAddr16,
  344. vpwAddrLen16,
  345. &addressLength,
  346. &pAddressLength,
  347. &realSockaddr)) {
  348. goto exit;
  349. }
  350. ul = GETWORD16( (*wsockapis[WOW_GETSOCKNAME].lpfn)( s32, realSockaddr, pAddressLength ) );
  351. // Note: 16-bit callbacks resulting from above function
  352. // call may have caused 16-bit memory movement
  353. FREEARGPTR(pFrame);
  354. FREEARGPTR(parg16);
  355. // Un-Thunk the 16-bit Address name and length buffers
  356. WSUnThunkAddrBufAndLen(ul,
  357. vpwAddrLen16,
  358. vpSockAddr16,
  359. addressLength,
  360. &fastSockaddr,
  361. realSockaddr);
  362. exit:
  363. FREEARGPTR( parg16 );
  364. RETURN( ul );
  365. } // WWS32getsockname
  366. ULONG FASTCALL WWS32getsockopt(PVDMFRAME pFrame)
  367. {
  368. ULONG ul = GETWORD16(INVALID_SOCKET);
  369. register PGETSOCKOPT16 parg16;
  370. SOCKET s32;
  371. WORD UNALIGNED *optionLength16;
  372. WORD actualOptionLength16;
  373. PBYTE optionValue16;
  374. DWORD optionLength32;
  375. PBYTE optionValue32;
  376. VPWORD vpwOptLen16;
  377. VPBYTE vpwOptVal16;
  378. WSEXIT_IF_NOT_INTIALIZED();
  379. GETARGPTR(pFrame, sizeof(PGETSOCKOPT16), parg16);
  380. //
  381. // Find the 32-bit socket handle.
  382. //
  383. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  384. goto exit;
  385. }
  386. vpwOptLen16 = (VPWORD)FETCHDWORD(parg16->OptionLength);
  387. vpwOptVal16 = (VPBYTE)FETCHDWORD(parg16->OptionValue);
  388. GETVDMPTR( vpwOptLen16, sizeof(WORD), optionLength16 );
  389. GETVDMPTR( vpwOptVal16, FETCHWORD(*optionLength16), optionValue16 );
  390. if ( FETCHWORD(*optionLength16) < sizeof(WORD) ) {
  391. FREEVDMPTR( optionLength16 );
  392. FREEVDMPTR( optionValue16 );
  393. FREEARGPTR( parg16 );
  394. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAEFAULT );
  395. ul = (ULONG)GETWORD16(SOCKET_ERROR );
  396. RETURN( ul );
  397. } else if ( FETCHWORD(*optionLength16) < sizeof(DWORD) ) {
  398. optionLength32 = sizeof(DWORD);
  399. } else {
  400. optionLength32 = FETCHWORD(*optionLength16);
  401. }
  402. optionValue32 = malloc_w(optionLength32);
  403. if ( optionValue32 == NULL ) {
  404. FREEVDMPTR( optionLength16 );
  405. FREEVDMPTR( optionValue16 );
  406. FREEARGPTR( parg16 );
  407. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAENOBUFS );
  408. ul = (ULONG)GETWORD16(SOCKET_ERROR );
  409. RETURN( ul );
  410. }
  411. SockCopyMemory( optionValue32, optionValue16, optionLength32 );
  412. ul = GETWORD16( (*wsockapis[WOW_GETSOCKOPT].lpfn)(
  413. s32,
  414. parg16->Level,
  415. SocketOption16To32( parg16->OptionName ),
  416. (char *)optionValue32,
  417. (int *)&optionLength32));
  418. // Note: 16-bit callbacks resulting from above function
  419. // call may have caused 16-bit memory movement
  420. FREEARGPTR(pFrame);
  421. FREEARGPTR(parg16);
  422. FREEVDMPTR(optionLength16);
  423. FREEVDMPTR(optionValue16);
  424. if ( ul == NO_ERROR ) {
  425. GETVDMPTR( vpwOptLen16, sizeof(WORD), optionLength16 );
  426. GETVDMPTR( vpwOptVal16, FETCHWORD(*optionLength16), optionValue16 );
  427. actualOptionLength16 = (WORD) min(optionLength32, FETCHWORD(*optionLength16));
  428. RtlMoveMemory( optionValue16, optionValue32, actualOptionLength16 );
  429. STOREWORD(*optionLength16, actualOptionLength16);
  430. FLUSHVDMPTR( vpwOptLen16, sizeof(parg16->OptionLength), optionLength16 );
  431. FLUSHVDMPTR( vpwOptVal16, actualOptionLength16, optionValue16 );
  432. }
  433. FREEVDMPTR( optionLength16 );
  434. FREEVDMPTR( optionValue16 );
  435. exit:
  436. FREEARGPTR( parg16 );
  437. RETURN( ul );
  438. } // WWS32getsockopt
  439. ULONG FASTCALL WWS32htonl(PVDMFRAME pFrame)
  440. {
  441. ULONG ul;
  442. register PHTONL16 parg16;
  443. GETARGPTR(pFrame, sizeof(HTONL16), parg16);
  444. ul = (*wsockapis[WOW_HTONL].lpfn)( parg16->HostLong );
  445. FREEARGPTR( parg16 );
  446. RETURN( ul );
  447. } // WWS32htonl
  448. ULONG FASTCALL WWS32htons(PVDMFRAME pFrame)
  449. {
  450. ULONG ul;
  451. register PHTONS16 parg16;
  452. GETARGPTR(pFrame, sizeof(HTONS16), parg16);
  453. ul = GETWORD16( (*wsockapis[WOW_HTONS].lpfn)( parg16->HostShort ) );
  454. FREEARGPTR( parg16 );
  455. RETURN( ul );
  456. } // WWS32htons
  457. ULONG FASTCALL WWS32inet_addr(PVDMFRAME pFrame)
  458. {
  459. ULONG ul;
  460. register PINET_ADDR16 parg16;
  461. PSZ addressString;
  462. CHAR szAddrStr[32];
  463. register PINET_ADDR16 realParg16;
  464. WSEXIT_IF_NOT_INTIALIZED();
  465. GETARGPTR(pFrame, sizeof(INET_ADDR16), parg16);
  466. realParg16 = parg16;
  467. GETVDMPTR( parg16->cp, 1, addressString );
  468. strcpy(szAddrStr, addressString);
  469. FREEVDMPTR( addressString );
  470. //
  471. // If the thread is version 1.0 of Windows Sockets, play special
  472. // stack games to return a struct in_addr.
  473. //
  474. if ( WWS32IsThreadVersion10 ) {
  475. PDWORD inAddr16;
  476. ULONG inAddr32;
  477. ul = *((PWORD)parg16);
  478. ul |= pFrame->wAppDS << 16;
  479. parg16 = (PINET_ADDR16)( (PCHAR)parg16 + 2 );
  480. inAddr32 = (*wsockapis[WOW_INET_ADDR].lpfn)( szAddrStr );
  481. ASSERT( sizeof(IN_ADDR) == sizeof(DWORD) );
  482. GETVDMPTR( ul, sizeof(DWORD), inAddr16 );
  483. STOREDWORD( *inAddr16, inAddr32 );
  484. FLUSHVDMPTR( ul, sizeof(DWORD), inAddr16 );
  485. FREEVDMPTR( inAddr16 );
  486. } else {
  487. ul = (*wsockapis[WOW_INET_ADDR].lpfn)( szAddrStr );
  488. }
  489. FREEARGPTR( realParg16 );
  490. RETURN( ul );
  491. } // WWS32inet_addr
  492. ULONG FASTCALL WWS32inet_ntoa(PVDMFRAME pFrame)
  493. {
  494. ULONG ul;
  495. register PINET_NTOA16 parg16;
  496. PSZ ipAddress;
  497. PSZ ipAddress16;
  498. IN_ADDR in32;
  499. WSEXIT_IF_NOT_INTIALIZED();
  500. GETARGPTR(pFrame, sizeof(INET_NTOA16), parg16);
  501. in32.s_addr = parg16->in;
  502. ipAddress = (PSZ) (*wsockapis[WOW_INET_NTOA].lpfn)( in32 );
  503. if ( ipAddress != NULL ) {
  504. GETVDMPTR( WWS32vIpAddress, strlen( ipAddress )+1, ipAddress16 );
  505. strcpy( ipAddress16, ipAddress );
  506. FLUSHVDMPTR( WWS32vIpAddress, strlen( ipAddress )+1, ipAddress16 );
  507. FREEVDMPTR( ipAddress16 );
  508. ul = WWS32vIpAddress;
  509. } else {
  510. ul = 0;
  511. }
  512. FREEARGPTR( parg16 );
  513. RETURN( ul );
  514. } // WWS32inet_ntoa
  515. ULONG FASTCALL WWS32ioctlsocket(PVDMFRAME pFrame)
  516. {
  517. ULONG ul;
  518. register PIOCTLSOCKET16 parg16;
  519. SOCKET s32;
  520. PDWORD argument16;
  521. DWORD argument32;
  522. DWORD command;
  523. VPDWORD vpdwArg16;
  524. WSEXIT_IF_NOT_INTIALIZED();
  525. GETARGPTR(pFrame, sizeof(IOCTLSOCKET16), parg16);
  526. //
  527. // Find the 32-bit socket handle.
  528. //
  529. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  530. goto exit;
  531. }
  532. vpdwArg16 = (VPDWORD)FETCHDWORD(parg16->Argument);
  533. GETVDMPTR( vpdwArg16, sizeof(*argument16), argument16 );
  534. //
  535. // Translate the command value as necessary.
  536. //
  537. switch ( FETCHDWORD( parg16->Command ) & IOCPARM_MASK ) {
  538. case 127:
  539. command = FIONREAD;
  540. break;
  541. case 126:
  542. command = FIONBIO;
  543. break;
  544. case 125:
  545. command = FIOASYNC;
  546. break;
  547. case 0:
  548. command = SIOCSHIWAT;
  549. break;
  550. case 1:
  551. command = SIOCGHIWAT;
  552. break;
  553. case 2:
  554. command = SIOCSLOWAT;
  555. break;
  556. case 3:
  557. command = SIOCGLOWAT;
  558. break;
  559. case 7:
  560. command = SIOCATMARK;
  561. break;
  562. default:
  563. command = 0;
  564. break;
  565. }
  566. argument32 = FETCHDWORD( *argument16 );
  567. ul = GETWORD16( (*wsockapis[WOW_IOCTLSOCKET].lpfn)(s32,
  568. command,
  569. &argument32));
  570. // Note: 16-bit callbacks resulting from above function
  571. // call may have caused 16-bit memory movement
  572. FREEARGPTR( parg16 );
  573. GETVDMPTR( vpdwArg16, sizeof(*argument16), argument16 );
  574. STOREDWORD( *argument16, argument32 );
  575. FLUSHVDMPTR( vpdwArg16, sizeof(*argument16), argument16 );
  576. FREEVDMPTR( argument16 );
  577. exit:
  578. FREEARGPTR( parg16 );
  579. RETURN( ul );
  580. } // WWS32ioctlsocket
  581. ULONG FASTCALL WWS32listen(PVDMFRAME pFrame)
  582. {
  583. ULONG ul;
  584. register PLISTEN16 parg16;
  585. SOCKET s32;
  586. WSEXIT_IF_NOT_INTIALIZED();
  587. GETARGPTR(pFrame, sizeof(PLISTEN6), parg16);
  588. //
  589. // Find the 32-bit socket handle.
  590. //
  591. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  592. goto exit;
  593. }
  594. ul = GETWORD16( (*wsockapis[WOW_LISTEN].lpfn)( s32, parg16->Backlog ) );
  595. exit:
  596. FREEARGPTR( parg16 );
  597. RETURN( ul );
  598. } // WWS32listen
  599. ULONG FASTCALL WWS32ntohl(PVDMFRAME pFrame)
  600. {
  601. ULONG ul;
  602. register PNTOHL16 parg16;
  603. GETARGPTR(pFrame, sizeof(NTOHL16), parg16);
  604. ul = (*wsockapis[WOW_NTOHL].lpfn)( parg16->NetLong );
  605. FREEARGPTR( parg16 );
  606. RETURN( ul );
  607. } // WWS32ntohl
  608. ULONG FASTCALL WWS32ntohs(PVDMFRAME pFrame)
  609. {
  610. ULONG ul;
  611. register PNTOHS16 parg16;
  612. GETARGPTR(pFrame, sizeof(NTOHS16), parg16);
  613. ul = GETWORD16( (*wsockapis[WOW_NTOHS].lpfn)( parg16->NetShort ) );
  614. FREEARGPTR( parg16 );
  615. RETURN( ul );
  616. } // WWS32ntohs
  617. ULONG FASTCALL WWS32recv(PVDMFRAME pFrame)
  618. {
  619. ULONG ul = GETWORD16(INVALID_SOCKET);
  620. register PRECV16 parg16;
  621. SOCKET s32;
  622. PBYTE buffer;
  623. INT BufferLength;
  624. VPBYTE vpBuf16;
  625. WSEXIT_IF_NOT_INTIALIZED();
  626. GETARGPTR(pFrame, sizeof(PRECV16), parg16);
  627. //
  628. // Find the 32-bit socket handle.
  629. //
  630. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  631. goto exit;
  632. }
  633. BufferLength = INT32(parg16->BufferLength);
  634. vpBuf16 = (VPBYTE)FETCHDWORD(parg16->Buffer);
  635. // Thunk the 16-bit recv buffer
  636. if(!WSThunkRecvBuffer(BufferLength, vpBuf16, &buffer)) {
  637. goto exit;
  638. }
  639. ul = GETWORD16( (*wsockapis[WOW_RECV].lpfn)(s32,
  640. buffer,
  641. BufferLength,
  642. parg16->Flags));
  643. // Note: 16-bit callbacks resulting from above function
  644. // call may have caused 16-bit memory movement
  645. FREEARGPTR(pFrame);
  646. FREEARGPTR(parg16);
  647. // Un-Thunk the 16-bit recv buffer
  648. WSUnthunkRecvBuffer((INT)ul, BufferLength, vpBuf16, buffer);
  649. exit:
  650. FREEARGPTR( parg16 );
  651. RETURN( ul );
  652. } // WWS32recv
  653. ULONG FASTCALL WWS32recvfrom(PVDMFRAME pFrame)
  654. {
  655. ULONG ul = GETWORD16(INVALID_SOCKET);
  656. register PRECVFROM16 parg16;
  657. SOCKET s32;
  658. INT addressLength;
  659. PINT pAddressLength;
  660. SOCKADDR fastSockaddr;
  661. PSOCKADDR realSockaddr;
  662. PBYTE buffer;
  663. INT BufferLength;
  664. VPBYTE vpBuf16;
  665. VPWORD vpwAddrLen16;
  666. VPSOCKADDR vpSockAddr16;
  667. WSEXIT_IF_NOT_INTIALIZED();
  668. GETARGPTR(pFrame, sizeof(PRECVFROM16), parg16);
  669. //
  670. // Find the 32-bit socket handle.
  671. //
  672. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  673. goto exit;
  674. }
  675. vpwAddrLen16 = (VPWORD)FETCHDWORD(parg16->AddressLength);
  676. vpSockAddr16 = (VPSOCKADDR)FETCHDWORD(parg16->Address);
  677. BufferLength = INT32(parg16->BufferLength);
  678. vpBuf16 = (VPBYTE)parg16->Buffer;
  679. // Thunk the 16-bit Address name and length buffers
  680. if(!WSThunkAddrBufAndLen(&fastSockaddr,
  681. vpSockAddr16,
  682. vpwAddrLen16,
  683. &addressLength,
  684. &pAddressLength,
  685. &realSockaddr)) {
  686. goto exit;
  687. }
  688. // Thunk the 16-bit recv buffer
  689. if(!WSThunkRecvBuffer(BufferLength, vpBuf16, &buffer)) {
  690. goto exit;
  691. }
  692. ul = GETWORD16( (*wsockapis[WOW_RECVFROM].lpfn)(s32,
  693. buffer,
  694. BufferLength,
  695. parg16->Flags,
  696. realSockaddr,
  697. pAddressLength));
  698. // Note: 16-bit callbacks resulting from above function
  699. // call may have caused 16-bit memory movement
  700. FREEARGPTR(pFrame);
  701. FREEARGPTR(parg16);
  702. // Un-Thunk the 16-bit Address name and length buffers
  703. WSUnThunkAddrBufAndLen(ul,
  704. vpwAddrLen16,
  705. vpSockAddr16,
  706. addressLength,
  707. &fastSockaddr,
  708. realSockaddr);
  709. // Un-Thunk the 16-bit recv buffer
  710. WSUnthunkRecvBuffer((INT)ul, BufferLength, vpBuf16, buffer);
  711. exit:
  712. FREEARGPTR( parg16 );
  713. RETURN( ul );
  714. } // WWS32recvfrom
  715. ULONG FASTCALL WWS32select(PVDMFRAME pFrame)
  716. {
  717. ULONG ul = (ULONG)GETWORD16( SOCKET_ERROR );
  718. register PSELECT16 parg16;
  719. PFD_SET readfds32 = NULL;
  720. PFD_SET writefds32 = NULL;
  721. PFD_SET exceptfds32 = NULL;
  722. PFD_SET16 readfds16;
  723. PFD_SET16 writefds16;
  724. PFD_SET16 exceptfds16;
  725. struct timeval timeout32;
  726. struct timeval *ptimeout32;
  727. PTIMEVAL16 timeout16;
  728. INT err;
  729. VPFD_SET16 vpreadfds16;
  730. VPFD_SET16 vpwritefds16;
  731. VPFD_SET16 vpexceptfds16;
  732. VPTIMEVAL16 vptimeout16;
  733. WSEXIT_IF_NOT_INTIALIZED();
  734. GETARGPTR( pFrame, sizeof(PSELECT16), parg16 );
  735. //
  736. // Get 16-bit pointers.
  737. //
  738. // !!! This sizeof(FD_SET16) here and below is wrong if the app is
  739. // using more than FDSETSIZE handles!!!
  740. vpreadfds16 = parg16->Readfds;
  741. vpwritefds16 = parg16->Writefds;
  742. vpexceptfds16 = parg16->Exceptfds;
  743. vptimeout16 = parg16->Timeout;
  744. GETOPTPTR(vpreadfds16, sizeof(FD_SET16), readfds16);
  745. GETOPTPTR(vpwritefds16, sizeof(FD_SET16), writefds16);
  746. GETOPTPTR(vpexceptfds16, sizeof(FD_SET16), exceptfds16);
  747. GETOPTPTR(vptimeout16, sizeof(TIMEVAL16), timeout16);
  748. //
  749. // Translate readfds.
  750. //
  751. if ( readfds16 != NULL ) {
  752. readfds32 = AllocateFdSet32( readfds16 );
  753. if ( readfds32 == NULL ) {
  754. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAENOBUFS );
  755. goto exit;
  756. }
  757. err = ConvertFdSet16To32( readfds16, readfds32 );
  758. if ( err != 0 ) {
  759. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( err );
  760. goto exit;
  761. }
  762. }
  763. //
  764. // Translate writefds.
  765. //
  766. if ( writefds16 != NULL ) {
  767. writefds32 = AllocateFdSet32( writefds16 );
  768. if ( writefds32 == NULL ) {
  769. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAENOBUFS );
  770. goto exit;
  771. }
  772. err = ConvertFdSet16To32( writefds16, writefds32 );
  773. if ( err != 0 ) {
  774. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( err );
  775. goto exit;
  776. }
  777. }
  778. //
  779. // Translate exceptfds.
  780. //
  781. if ( exceptfds16 != NULL ) {
  782. exceptfds32 = AllocateFdSet32( exceptfds16 );
  783. if ( exceptfds32 == NULL ) {
  784. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAENOBUFS );
  785. goto exit;
  786. }
  787. err = ConvertFdSet16To32( exceptfds16, exceptfds32 );
  788. if ( err != 0 ) {
  789. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( err );
  790. goto exit;
  791. }
  792. }
  793. //
  794. // Translate the timeout.
  795. //
  796. if ( timeout16 == NULL ) {
  797. ptimeout32 = NULL;
  798. } else {
  799. timeout32.tv_sec = FETCHDWORD( timeout16->tv_sec );
  800. timeout32.tv_usec = FETCHDWORD( timeout16->tv_usec );
  801. ptimeout32 = &timeout32;
  802. }
  803. //
  804. // Call the 32-bit select function.
  805. //
  806. ul = GETWORD16( (*wsockapis[WOW_SELECT].lpfn)(0,
  807. readfds32,
  808. writefds32,
  809. exceptfds32,
  810. ptimeout32));
  811. // Note: 16-bit callbacks resulting from above function
  812. // call may have caused 16-bit memory movement
  813. FREEARGPTR(pFrame);
  814. FREEARGPTR(parg16);
  815. FREEOPTPTR(readfds16);
  816. FREEOPTPTR(writefds16);
  817. FREEOPTPTR(exceptfds16);
  818. FREEOPTPTR(timeout16);
  819. //
  820. // Copy 32-bit readfds back to the 16-bit readfds.
  821. //
  822. if ( readfds32 != NULL ) {
  823. GETOPTPTR(vpreadfds16, sizeof(FD_SET16), readfds16);
  824. ConvertFdSet32To16( readfds32, readfds16 );
  825. FLUSHVDMPTR(vpreadfds16, sizeof(FD_SET16), readfds16);
  826. }
  827. //
  828. // Copy 32-bit writefds back to the 16-bit writefds.
  829. //
  830. if ( writefds32 != NULL ) {
  831. GETOPTPTR(vpwritefds16, sizeof(FD_SET16), writefds16);
  832. ConvertFdSet32To16( writefds32, writefds16 );
  833. FLUSHVDMPTR(vpwritefds16, sizeof(FD_SET16), writefds16);
  834. }
  835. //
  836. // Copy 32-bit exceptfds back to the 16-bit exceptfds.
  837. //
  838. if ( exceptfds32 != NULL ) {
  839. GETOPTPTR(vpexceptfds16, sizeof(FD_SET16), exceptfds16);
  840. ConvertFdSet32To16( exceptfds32, exceptfds16 );
  841. FLUSHVDMPTR(vpexceptfds16, sizeof(FD_SET16), exceptfds16);
  842. }
  843. exit:
  844. FREEOPTPTR( readfds16 );
  845. FREEOPTPTR( writefds16 );
  846. FREEOPTPTR( exceptfds16 );
  847. if ( readfds32 != NULL ) {
  848. free_w((PVOID)readfds32);
  849. }
  850. if ( writefds32 != NULL ) {
  851. free_w((PVOID)writefds32);
  852. }
  853. if ( exceptfds32 != NULL ) {
  854. free_w((PVOID)exceptfds32);
  855. }
  856. FREEARGPTR( parg16 );
  857. RETURN( ul );
  858. } // WWS32select
  859. ULONG FASTCALL WWS32send(PVDMFRAME pFrame)
  860. {
  861. ULONG ul = GETWORD16(INVALID_SOCKET);
  862. register PSEND16 parg16;
  863. SOCKET s32;
  864. INT BufferLength;
  865. PBYTE buffer;
  866. VPBYTE vpBuf16;
  867. WSEXIT_IF_NOT_INTIALIZED();
  868. GETARGPTR(pFrame, sizeof(PSEND16), parg16);
  869. //
  870. // Find the 32-bit socket handle.
  871. //
  872. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  873. goto exit;
  874. }
  875. BufferLength = INT32(parg16->BufferLength);
  876. vpBuf16 = FETCHDWORD(parg16->Buffer);
  877. // Thunk the 16-bit send buffer
  878. if(!WSThunkSendBuffer(BufferLength, vpBuf16, &buffer)) {
  879. goto exit;
  880. }
  881. ul = GETWORD16( (*wsockapis[WOW_SEND].lpfn)(s32,
  882. buffer,
  883. BufferLength,
  884. parg16->Flags));
  885. // Note: 16-bit callbacks resulting from above function
  886. // call may have caused 16-bit memory movement
  887. FREEARGPTR(pFrame);
  888. FREEARGPTR(parg16);
  889. // Un-Thunk the 16-bit send buffer
  890. WSUnthunkSendBuffer(buffer);
  891. exit:
  892. FREEARGPTR( parg16 );
  893. RETURN( ul );
  894. } // WWS32send
  895. ULONG FASTCALL WWS32sendto(PVDMFRAME pFrame)
  896. {
  897. ULONG ul = GETWORD16(INVALID_SOCKET);
  898. register PSENDTO16 parg16;
  899. SOCKET s32;
  900. PBYTE buffer;
  901. SOCKADDR fastSockaddr;
  902. PSOCKADDR realSockaddr;
  903. INT addressLength;
  904. INT BufferLength;
  905. VPSOCKADDR vpSockAddr16;
  906. VPBYTE vpBuf16;
  907. WSEXIT_IF_NOT_INTIALIZED();
  908. GETARGPTR(pFrame, sizeof(PSENDTO16), parg16);
  909. //
  910. // Find the 32-bit socket handle.
  911. //
  912. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  913. goto exit;
  914. }
  915. addressLength = INT32(parg16->AddressLength);
  916. vpSockAddr16 = (VPSOCKADDR)FETCHDWORD(parg16->Address);
  917. BufferLength = INT32(parg16->BufferLength);
  918. vpBuf16 = (VPBYTE)FETCHDWORD(parg16->Buffer);
  919. // Thunk the 16-bit Address buffer
  920. if(!WSThunkAddrBuf(addressLength,
  921. vpSockAddr16,
  922. &fastSockaddr,
  923. &realSockaddr)) {
  924. goto exit;
  925. }
  926. // Thunk the 16-bit send buffer
  927. if(!WSThunkSendBuffer(BufferLength, vpBuf16, &buffer)) {
  928. goto exit;
  929. }
  930. ul = GETWORD16( (*wsockapis[WOW_SENDTO].lpfn)(s32,
  931. buffer,
  932. BufferLength,
  933. parg16->Flags,
  934. realSockaddr,
  935. addressLength));
  936. // Note: 16-bit callbacks resulting from above function
  937. // call may have caused 16-bit memory movement
  938. FREEARGPTR(pFrame);
  939. FREEARGPTR(parg16);
  940. // Un-Thunk the 16-bit address buffer
  941. WSUnThunkAddrBuf(&fastSockaddr, realSockaddr);
  942. // Un-Thunk the 16-bit send buffer
  943. WSUnthunkSendBuffer(buffer);
  944. exit:
  945. FREEARGPTR( parg16 );
  946. RETURN( ul );
  947. } // WWS32sendto
  948. ULONG FASTCALL WWS32setsockopt(PVDMFRAME pFrame)
  949. {
  950. ULONG ul = GETWORD16(INVALID_SOCKET);
  951. register PSETSOCKOPT16 parg16;
  952. SOCKET s32;
  953. PBYTE optionValue16;
  954. PBYTE optionValue32;
  955. DWORD optionLength32;
  956. WSEXIT_IF_NOT_INTIALIZED();
  957. GETARGPTR(pFrame, sizeof(PSETSOCKOPT16), parg16);
  958. //
  959. // Find the 32-bit socket handle.
  960. //
  961. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  962. goto exit;
  963. }
  964. GETVDMPTR( parg16->OptionValue, parg16->OptionLength, optionValue16 );
  965. if ( parg16->OptionLength < sizeof(DWORD) ) {
  966. optionLength32 = sizeof(DWORD);
  967. } else {
  968. optionLength32 = parg16->OptionLength;
  969. }
  970. optionValue32 = malloc_w(optionLength32);
  971. if ( optionValue32 == NULL ) {
  972. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAENOBUFS );
  973. ul = (ULONG)GETWORD16( SOCKET_ERROR );
  974. FREEVDMPTR( optionValue16 );
  975. FREEARGPTR( parg16 );
  976. RETURN( ul );
  977. }
  978. RtlZeroMemory( optionValue32, optionLength32 );
  979. RtlMoveMemory( optionValue32, optionValue16, parg16->OptionLength );
  980. ul = GETWORD16( (*wsockapis[WOW_SETSOCKOPT].lpfn)(
  981. s32,
  982. parg16->Level,
  983. SocketOption16To32( parg16->OptionName ),
  984. optionValue32,
  985. optionLength32));
  986. // Note: 16-bit callbacks resulting from above function
  987. // call may have caused 16-bit memory movement
  988. FREEARGPTR(pFrame);
  989. FREEARGPTR(parg16);
  990. FREEVDMPTR( optionValue16 );
  991. free_w(optionValue32);
  992. exit:
  993. FREEARGPTR( parg16 );
  994. RETURN( ul );
  995. } // WWS32setsockopt
  996. ULONG FASTCALL WWS32shutdown(PVDMFRAME pFrame)
  997. {
  998. ULONG ul = GETWORD16(INVALID_SOCKET);
  999. register PSHUTDOWN16 parg16;
  1000. SOCKET s32;
  1001. WSEXIT_IF_NOT_INTIALIZED();
  1002. GETARGPTR(pFrame, sizeof(PBIND16), parg16);
  1003. //
  1004. // Find the 32-bit socket handle.
  1005. //
  1006. if((s32 = WSGetWinsock32(parg16->hSocket, &ul)) == INVALID_SOCKET) {
  1007. goto exit;
  1008. }
  1009. ul = GETWORD16( (*wsockapis[WOW_SHUTDOWN].lpfn)( s32, parg16->How ) );
  1010. exit:
  1011. FREEARGPTR( parg16 );
  1012. RETURN( ul );
  1013. } // WWS32shutdown
  1014. ULONG FASTCALL WWS32socket(PVDMFRAME pFrame)
  1015. {
  1016. ULONG ul = GETWORD16(INVALID_SOCKET);
  1017. register PSOCKET16 parg16;
  1018. SOCKET s32;
  1019. HSOCKET16 s16;
  1020. WSEXIT_IF_NOT_INTIALIZED();
  1021. GETARGPTR(pFrame, sizeof(SOCKET16), parg16);
  1022. s32 = (*wsockapis[WOW_SOCKET].lpfn)(INT32(parg16->AddressFamily),
  1023. INT32(parg16->Type),
  1024. INT32(parg16->Protocol));
  1025. // Note: 16-bit callbacks resulting from above function
  1026. // call may have caused 16-bit memory movement
  1027. FREEARGPTR(pFrame);
  1028. FREEARGPTR(parg16);
  1029. //
  1030. // If the call succeeded, alias the 32-bit socket handle we just
  1031. // obtained into a 16-bit handle.
  1032. //
  1033. if ( s32 != INVALID_SOCKET ) {
  1034. s16 = GetWinsock16( s32, 0 );
  1035. ul = s16;
  1036. if ( s16 == 0 ) {
  1037. (*wsockapis[WOW_CLOSESOCKET].lpfn)( s32 );
  1038. (*wsockapis[WOW_WSASETLASTERROR].lpfn)( WSAENOBUFS );
  1039. ul = GETWORD16( INVALID_SOCKET );
  1040. }
  1041. } else {
  1042. ul = GETWORD16( INVALID_SOCKET );
  1043. }
  1044. FREEARGPTR( parg16 );
  1045. RETURN( ul );
  1046. } // WWS32socket
  1047. //
  1048. // Routines for converting between 16- and 32-bit FD_SET structures.
  1049. //
  1050. PFD_SET AllocateFdSet32(IN PFD_SET16 FdSet16)
  1051. {
  1052. int bytes = 4 + (FETCHWORD(FdSet16->fd_count) * sizeof(SOCKET));
  1053. return (PFD_SET)( malloc_w(bytes) );
  1054. } // AlloacteFdSet32
  1055. INT ConvertFdSet16To32(IN PFD_SET16 FdSet16,
  1056. IN PFD_SET FdSet32)
  1057. {
  1058. int i;
  1059. FdSet32->fd_count = UINT32( FdSet16->fd_count );
  1060. for ( i = 0; i < (int)FdSet32->fd_count; i++ ) {
  1061. FdSet32->fd_array[i] = GetWinsock32( FdSet16->fd_array[i] );
  1062. if ( FdSet32->fd_array[i] == INVALID_SOCKET ) {
  1063. return WSAENOTSOCK;
  1064. }
  1065. }
  1066. return 0;
  1067. } // ConvertFdSet16To32
  1068. VOID ConvertFdSet32To16(IN PFD_SET FdSet32,
  1069. IN PFD_SET16 FdSet16)
  1070. {
  1071. int i;
  1072. STOREWORD( FdSet16->fd_count, GETWORD16( FdSet32->fd_count ) );
  1073. for ( i = 0; i < FdSet16->fd_count; i++ ) {
  1074. HSOCKET16 s16;
  1075. s16 = GetWinsock16( FdSet32->fd_array[i], 0 );
  1076. STOREWORD( FdSet16->fd_array[i], s16 );
  1077. }
  1078. } // ConvertFdSet32To16
  1079. //
  1080. // Routines for aliasing 32-bit socket handles to 16-bit handles.
  1081. //
  1082. PWINSOCK_SOCKET_INFO FindSocketInfo16(IN SOCKET h32,
  1083. IN HAND16 h16)
  1084. {
  1085. PLIST_ENTRY listEntry;
  1086. PWINSOCK_SOCKET_INFO socketInfo;
  1087. //
  1088. // It is the responsibility of the caller of this routine to enter
  1089. // the critical section that protects the global socket list.
  1090. //
  1091. for ( listEntry = WWS32SocketHandleListHead.Flink;
  1092. listEntry != &WWS32SocketHandleListHead;
  1093. listEntry = listEntry->Flink ) {
  1094. socketInfo = CONTAINING_RECORD(listEntry,
  1095. WINSOCK_SOCKET_INFO,
  1096. GlobalSocketListEntry);
  1097. if ( socketInfo->SocketHandle32 == h32 ||
  1098. socketInfo->SocketHandle16 == h16 ) {
  1099. return socketInfo;
  1100. }
  1101. }
  1102. return NULL;
  1103. } // FindSocketInfo16
  1104. HAND16 AllocateUnique16BitHandle(VOID)
  1105. {
  1106. PLIST_ENTRY listEntry;
  1107. PWINSOCK_SOCKET_INFO socketInfo;
  1108. HAND16 h16;
  1109. WORD i;
  1110. //
  1111. // This function assumes it is called with the WWS32CriticalSection
  1112. // lock held!
  1113. //
  1114. //
  1115. // If the socket list is empty, then we can reset our socket handle
  1116. // counter because we know there are no active sockets. We'll only
  1117. // do this if the handle counter is above some value (just pulled
  1118. // out of the air) so that handles are not reused too quickly.
  1119. // (Frequent handle reuse can confuse poorly written 16-bit apps.)
  1120. //
  1121. if( ( WWS32SocketHandleCounter > 255 ) &&
  1122. IsListEmpty( &WWS32SocketHandleListHead ) ) {
  1123. WWS32SocketHandleCounter = 1;
  1124. WWS32SocketHandleCounterWrapped = FALSE;
  1125. }
  1126. //
  1127. // If the socket handle counter has not wrapped around,
  1128. // then we can quickly return a unique handle.
  1129. //
  1130. if( !WWS32SocketHandleCounterWrapped ) {
  1131. h16 = (HAND16)WWS32SocketHandleCounter++;
  1132. if( WWS32SocketHandleCounter == 0xFFFF ) {
  1133. WWS32SocketHandleCounter = 1;
  1134. WWS32SocketHandleCounterWrapped = TRUE;
  1135. }
  1136. ASSERT( h16 != 0 );
  1137. return h16;
  1138. }
  1139. //
  1140. // There are active sockets, and the socket handle counter has
  1141. // wrapped, so we'll need to perform a painful search for a unique
  1142. // handle. We'll put a cap on the maximum number of times through
  1143. // this search loop so that, if all handles from 1 to 0xFFFE are
  1144. // in use, we won't search forever for something we'll never find.
  1145. //
  1146. for( i = 1 ; i <= 0xFFFE ; i++ ) {
  1147. h16 = (HAND16)WWS32SocketHandleCounter++;
  1148. if( WWS32SocketHandleCounter == 0xFFFF ) {
  1149. WWS32SocketHandleCounter = 1;
  1150. }
  1151. for ( listEntry = WWS32SocketHandleListHead.Flink;
  1152. listEntry != &WWS32SocketHandleListHead;
  1153. listEntry = listEntry->Flink ) {
  1154. socketInfo = CONTAINING_RECORD(
  1155. listEntry,
  1156. WINSOCK_SOCKET_INFO,
  1157. GlobalSocketListEntry
  1158. );
  1159. if( socketInfo->SocketHandle16 == h16 ) {
  1160. break;
  1161. }
  1162. }
  1163. //
  1164. // If listEntry == &WWS32SocketHandleListHead, then we have
  1165. // scanned the entire list and found no match. This is good,
  1166. // and we'll just return the current handle. Otherwise, there
  1167. // was a collision, so we'll get another potential handle and
  1168. // rescan the list.
  1169. //
  1170. if( listEntry == &WWS32SocketHandleListHead ) {
  1171. ASSERT( h16 != 0 );
  1172. return h16;
  1173. }
  1174. }
  1175. //
  1176. // If we made it this far, then there were no unique handles
  1177. // available. Bad news.
  1178. //
  1179. return 0;
  1180. } // AllocateUnique16BitHandle
  1181. HAND16 GetWinsock16(IN INT h32,
  1182. IN INT iClass)
  1183. {
  1184. PWINSOCK_SOCKET_INFO socketInfo;
  1185. HAND16 h16;
  1186. RtlEnterCriticalSection( &WWS32CriticalSection );
  1187. //
  1188. // If the handle is already in the list, use it.
  1189. //
  1190. socketInfo = FindSocketInfo16( h32, 0 );
  1191. if ( socketInfo != NULL ) {
  1192. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1193. return socketInfo->SocketHandle16;
  1194. }
  1195. //
  1196. // If this thread has not yet been initialized, then we cannot
  1197. // create the new socket data. This should only happen if a 16-bit
  1198. // app closes a socket while an async connect is outstanding.
  1199. //
  1200. if( !WWS32IsThreadInitialized ) {
  1201. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1202. return 0;
  1203. }
  1204. //
  1205. // The handle is not in use. Create a new entry in the list.
  1206. //
  1207. h16 = AllocateUnique16BitHandle();
  1208. if( h16 == 0 ) {
  1209. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1210. return 0;
  1211. }
  1212. socketInfo = malloc_w(sizeof(*socketInfo));
  1213. if ( socketInfo == NULL ) {
  1214. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1215. return 0;
  1216. }
  1217. socketInfo->SocketHandle16 = h16;
  1218. socketInfo->SocketHandle32 = h32;
  1219. socketInfo->ThreadSerialNumber = WWS32ThreadSerialNumber;
  1220. InsertTailList( &WWS32SocketHandleListHead, &socketInfo->GlobalSocketListEntry );
  1221. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1222. ASSERT( h16 != 0 );
  1223. return h16;
  1224. } // GetWinsock16
  1225. VOID FreeWinsock16(IN HAND16 h16)
  1226. {
  1227. PWINSOCK_SOCKET_INFO socketInfo;
  1228. RtlEnterCriticalSection( &WWS32CriticalSection );
  1229. socketInfo = FindSocketInfo16( INVALID_SOCKET, h16 );
  1230. if ( socketInfo == NULL ) {
  1231. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1232. return;
  1233. }
  1234. RemoveEntryList( &socketInfo->GlobalSocketListEntry );
  1235. free_w((PVOID)socketInfo);
  1236. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1237. return;
  1238. } // FreeWinsock16
  1239. DWORD GetWinsock32(IN HAND16 h16)
  1240. {
  1241. PWINSOCK_SOCKET_INFO socketInfo;
  1242. SOCKET socket32;
  1243. RtlEnterCriticalSection( &WWS32CriticalSection );
  1244. socketInfo = FindSocketInfo16( INVALID_SOCKET, h16 );
  1245. if ( socketInfo == NULL ) {
  1246. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1247. return INVALID_SOCKET;
  1248. }
  1249. //
  1250. // Store the socket handle in an aytumatic before leaving the critical
  1251. // section in case the socketInfo structure is about to be freed.
  1252. //
  1253. socket32 = socketInfo->SocketHandle32;
  1254. RtlLeaveCriticalSection( &WWS32CriticalSection );
  1255. return socket32;
  1256. } // GetWinsock32
  1257. int SocketOption16To32(IN WORD SocketOption16)
  1258. {
  1259. if ( SocketOption16 == 0xFF7F ) {
  1260. return SO_DONTLINGER;
  1261. }
  1262. return (int)SocketOption16;
  1263. } // SocketOption16To32
  1264. DWORD WSGetWinsock32 (IN HAND16 h16,
  1265. OUT PULONG pul)
  1266. {
  1267. DWORD s32;
  1268. s32 = GetWinsock32(h16);
  1269. if(s32 == INVALID_SOCKET) {
  1270. (*wsockapis[WOW_WSASETLASTERROR].lpfn)(WSAENOTSOCK);
  1271. *pul = (ULONG)GETWORD16(SOCKET_ERROR);
  1272. }
  1273. return(s32);
  1274. }
  1275. BOOL WSThunkAddrBufAndLen(IN PSOCKADDR fastSockaddr,
  1276. IN VPSOCKADDR vpSockAddr16,
  1277. IN VPWORD vpwAddrLen16,
  1278. OUT PINT addressLength,
  1279. OUT PINT *pAddressLength,
  1280. OUT PSOCKADDR *realSockaddr)
  1281. {
  1282. PWORD addressLength16;
  1283. PSOCKADDR Sockaddr;
  1284. GETVDMPTR(vpwAddrLen16, sizeof(*addressLength16), addressLength16);
  1285. GETVDMPTR(vpSockAddr16, *addressLength16, Sockaddr);
  1286. if(Sockaddr) {
  1287. *realSockaddr = fastSockaddr;
  1288. }
  1289. else {
  1290. *realSockaddr = NULL;
  1291. }
  1292. if (addressLength16 == NULL) {
  1293. *pAddressLength = NULL;
  1294. } else {
  1295. *addressLength = INT32(*addressLength16);
  1296. *pAddressLength = addressLength;
  1297. if(*addressLength > sizeof(SOCKADDR)) {
  1298. *realSockaddr = malloc_w(*addressLength);
  1299. if(*realSockaddr == NULL) {
  1300. (*wsockapis[WOW_WSASETLASTERROR].lpfn)(WSAENOBUFS);
  1301. return(FALSE);
  1302. }
  1303. }
  1304. }
  1305. FREEVDMPTR(Sockaddr);
  1306. FREEVDMPTR(addressLength16);
  1307. return(TRUE);
  1308. }
  1309. VOID WSUnThunkAddrBufAndLen(IN ULONG ret,
  1310. IN VPWORD vpwAddrLen16,
  1311. IN VPSOCKADDR vpSockAddr16,
  1312. IN INT addressLength,
  1313. IN PSOCKADDR fastSockaddr,
  1314. IN PSOCKADDR realSockaddr)
  1315. {
  1316. PWORD addressLength16;
  1317. PSOCKADDR Sockaddr;
  1318. GETVDMPTR(vpwAddrLen16, sizeof(*addressLength16), addressLength16);
  1319. if((ret != SOCKET_ERROR) && addressLength16) {
  1320. STOREWORD(*addressLength16, addressLength);
  1321. FLUSHVDMPTR(vpwAddrLen16, sizeof(WORD), addressLength16);
  1322. GETVDMPTR(vpSockAddr16, addressLength, Sockaddr);
  1323. if(Sockaddr) {
  1324. // don't copy back to the 16-bit address buffer if it's too small
  1325. if(addressLength <= *addressLength16) {
  1326. SockCopyMemory(Sockaddr, realSockaddr, addressLength);
  1327. FLUSHVDMPTR(vpSockAddr16, addressLength, Sockaddr);
  1328. }
  1329. }
  1330. }
  1331. if( (realSockaddr) && (realSockaddr != fastSockaddr) ) {
  1332. free_w(realSockaddr);
  1333. }
  1334. FREEVDMPTR(addressLength16);
  1335. FREEVDMPTR(Sockaddr);
  1336. }
  1337. BOOL WSThunkAddrBuf(IN INT addressLength,
  1338. IN VPSOCKADDR vpSockAddr16,
  1339. IN PSOCKADDR fastSockaddr,
  1340. OUT PSOCKADDR *realSockaddr)
  1341. {
  1342. PSOCKADDR Sockaddr;
  1343. GETVDMPTR(vpSockAddr16, addressLength, Sockaddr);
  1344. if(Sockaddr) {
  1345. if(addressLength <= sizeof(SOCKADDR)) {
  1346. *realSockaddr = fastSockaddr;
  1347. }
  1348. else {
  1349. *realSockaddr = malloc_w(addressLength);
  1350. if(*realSockaddr == NULL) {
  1351. (*wsockapis[WOW_WSASETLASTERROR].lpfn)(WSAENOBUFS);
  1352. FREEVDMPTR(Sockaddr);
  1353. return(FALSE);
  1354. }
  1355. }
  1356. SockCopyMemory(*realSockaddr, Sockaddr, addressLength);
  1357. }
  1358. else {
  1359. *realSockaddr = NULL;
  1360. }
  1361. FREEVDMPTR(Sockaddr);
  1362. return(TRUE);
  1363. }
  1364. VOID WSUnThunkAddrBuf(IN PSOCKADDR fastSockaddr,
  1365. IN PSOCKADDR realSockaddr)
  1366. {
  1367. if( (realSockaddr) && (realSockaddr != fastSockaddr) ) {
  1368. free_w(realSockaddr);
  1369. }
  1370. }
  1371. BOOL WSThunkRecvBuffer(IN INT BufferLength,
  1372. IN VPBYTE vpBuf16,
  1373. OUT PBYTE *buffer)
  1374. {
  1375. PBYTE lpBuf16;
  1376. GETVDMPTR(vpBuf16, BufferLength, lpBuf16);
  1377. if(lpBuf16) {
  1378. *buffer = malloc_w(BufferLength);
  1379. if(*buffer == NULL) {
  1380. (*wsockapis[WOW_WSASETLASTERROR].lpfn)(WSAENOBUFS);
  1381. return(FALSE);
  1382. }
  1383. }
  1384. else {
  1385. *buffer = NULL;
  1386. }
  1387. return(TRUE);
  1388. }
  1389. VOID WSUnthunkRecvBuffer(IN INT cBytes,
  1390. IN INT BufferLength,
  1391. IN VPBYTE vpBuf16,
  1392. IN PBYTE buffer)
  1393. {
  1394. PBYTE lpBuf16;
  1395. GETVDMPTR(vpBuf16, BufferLength, lpBuf16);
  1396. if(buffer) {
  1397. if( (cBytes > 0) && lpBuf16 ) {
  1398. SockCopyMemory(lpBuf16, buffer, cBytes);
  1399. FLUSHVDMPTR(vpBuf16, cBytes, lpBuf16);
  1400. }
  1401. free_w(buffer);
  1402. }
  1403. FREEVDMPTR(lpBuf16);
  1404. }
  1405. BOOL WSThunkSendBuffer(IN INT BufferLength,
  1406. IN VPBYTE vpBuf16,
  1407. OUT PBYTE *buffer)
  1408. {
  1409. PBYTE lpBuf16;
  1410. GETVDMPTR(vpBuf16, BufferLength, lpBuf16);
  1411. if(lpBuf16) {
  1412. *buffer = malloc_w(BufferLength);
  1413. if(*buffer) {
  1414. SockCopyMemory(*buffer, lpBuf16, BufferLength);
  1415. }
  1416. else {
  1417. (*wsockapis[WOW_WSASETLASTERROR].lpfn)(WSAENOBUFS);
  1418. return(FALSE);
  1419. }
  1420. FREEVDMPTR(lpBuf16);
  1421. }
  1422. else {
  1423. *buffer = NULL;
  1424. }
  1425. return(TRUE);
  1426. }
  1427. VOID WSUnthunkSendBuffer(IN PBYTE buffer)
  1428. {
  1429. if(buffer) {
  1430. free_w(buffer);
  1431. }
  1432. }