Implemented DllCanUnloadNow.
[wine/multimedia.git] / dlls / winsock / protocol.c
blobeda84d8d91f38c290effcb67fdeefdac5726bb87
1 /*
2 * Protocol enumeration functions
4 * Copyright (C) 2001 Stefan Leichter
5 * Copyright (C) 2004 Hans Leidekker
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 /* 02/11/2004
23 * The protocol enumeration functions were verified to match Win2k versions
24 * for these protocols: IPX, SPX, SPXII, TCP/IP and UDP/IP.
27 #include "config.h"
29 #include <stdarg.h>
30 #include <stdio.h>
31 #include <string.h>
33 #include <sys/types.h>
34 #ifdef HAVE_ARPA_INET_H
35 #include <arpa/inet.h>
36 #endif
37 #ifdef HAVE_NETDB_H
38 #include <netdb.h>
39 #endif
41 #include "windef.h"
42 #include "winbase.h"
43 #include "winnls.h"
44 #include "wtypes.h"
45 #include "nspapi.h"
46 #include "winsock2.h"
47 #include "wsipx.h"
48 #include "wshisotp.h"
50 #include "wine/unicode.h"
51 #include "wine/debug.h"
53 WINE_DEFAULT_DEBUG_CHANNEL(winsock);
55 /* names of the protocols */
56 static const CHAR NameIpx[] = "IPX";
57 static const CHAR NameSpx[] = "SPX";
58 static const CHAR NameSpxII[] = "SPX II";
59 static const CHAR NameTcp[] = "TCP/IP";
60 static const CHAR NameUdp[] = "UDP/IP";
62 static const WCHAR NameIpxW[] = {'I', 'P', 'X', '\0'};
63 static const WCHAR NameSpxW[] = {'S', 'P', 'X', '\0'};
64 static const WCHAR NameSpxIIW[] = {'S', 'P', 'X', ' ', 'I', 'I', '\0'};
65 static const WCHAR NameTcpW[] = {'T', 'C', 'P', '/', 'I', 'P', '\0'};
66 static const WCHAR NameUdpW[] = {'U', 'D', 'P', '/', 'I', 'P', '\0'};
68 /* Taken from Win2k */
69 static const GUID ProviderIdIP = { 0xe70f1aa0, 0xab8b, 0x11cf,
70 { 0x8c, 0xa3, 0x00, 0x80, 0x5f, 0x48, 0xa1, 0x92 } };
71 static const GUID ProviderIdIPX = { 0x11058240, 0xbe47, 0x11cf,
72 { 0x95, 0xc8, 0x00, 0x80, 0x5f, 0x48, 0xa1, 0x92 } };
73 static const GUID ProviderIdSPX = { 0x11058241, 0xbe47, 0x11cf,
74 { 0x95, 0xc8, 0x00, 0x80, 0x5f, 0x48, 0xa1, 0x92 } };
76 /*****************************************************************************
77 * WINSOCK_EnterSingleProtocolW [internal]
79 * enters the protocol information of one given protocol into the given
80 * buffer.
82 * RETURNS
83 * 1 if a protocol was entered into the buffer.
84 * SOCKET_ERROR otherwise.
86 * BUGS
87 * - only implemented for IPX, SPX, SPXII, TCP, UDP
88 * - there is no check that the operating system supports the returned
89 * protocols
91 static INT WINSOCK_EnterSingleProtocolW( INT protocol, WSAPROTOCOL_INFOW* info )
93 memset( info, 0, sizeof(WSAPROTOCOL_INFOW) );
94 info->iProtocol = protocol;
96 switch (protocol)
98 case WS_IPPROTO_TCP:
99 info->dwServiceFlags1 = XP1_PARTIAL_MESSAGE | XP1_EXPEDITED_DATA |
100 XP1_GRACEFUL_CLOSE | XP1_GUARANTEED_ORDER |
101 XP1_GUARANTEED_DELIVERY;
102 memcpy( &info->ProviderId, &ProviderIdIP, sizeof(GUID) );
103 info->dwCatalogEntryId = 0x3e9;
104 info->ProtocolChain.ChainLen = 1;
105 info->iVersion = 2;
106 info->iAddressFamily = WS_AF_INET;
107 info->iMaxSockAddr = 0x10;
108 info->iMinSockAddr = 0x10;
109 info->iSocketType = WS_SOCK_STREAM;
110 strcpyW( info->szProtocol, NameTcpW );
111 break;
113 case WS_IPPROTO_UDP:
114 info->dwServiceFlags1 = XP1_PARTIAL_MESSAGE | XP1_SUPPORT_BROADCAST |
115 XP1_SUPPORT_MULTIPOINT | XP1_MESSAGE_ORIENTED |
116 XP1_CONNECTIONLESS;
117 memcpy( &info->ProviderId, &ProviderIdIP, sizeof(GUID) );
118 info->dwCatalogEntryId = 0x3ea;
119 info->ProtocolChain.ChainLen = 1;
120 info->iVersion = 2;
121 info->iAddressFamily = WS_AF_INET;
122 info->iMaxSockAddr = 0x10;
123 info->iMinSockAddr = 0x10;
124 info->iSocketType = WS_SOCK_DGRAM;
125 info->dwMessageSize = 0xffbb;
126 strcpyW( info->szProtocol, NameUdpW );
127 break;
129 case NSPROTO_IPX:
130 info->dwServiceFlags1 = XP1_PARTIAL_MESSAGE | XP1_SUPPORT_BROADCAST |
131 XP1_SUPPORT_MULTIPOINT | XP1_MESSAGE_ORIENTED |
132 XP1_CONNECTIONLESS;
133 memcpy( &info->ProviderId, &ProviderIdIPX, sizeof(GUID) );
134 info->dwCatalogEntryId = 0x406;
135 info->ProtocolChain.ChainLen = 1;
136 info->iVersion = 2;
137 info->iAddressFamily = WS_AF_IPX;
138 info->iMaxSockAddr = 0x10;
139 info->iMinSockAddr = 0x0e;
140 info->iSocketType = WS_SOCK_DGRAM;
141 info->iProtocolMaxOffset = 0xff;
142 info->dwMessageSize = 0x240;
143 strcpyW( info->szProtocol, NameIpxW );
144 break;
146 case NSPROTO_SPX:
147 info->dwServiceFlags1 = XP1_IFS_HANDLES | XP1_PSEUDO_STREAM |
148 XP1_MESSAGE_ORIENTED | XP1_GUARANTEED_ORDER |
149 XP1_GUARANTEED_DELIVERY;
150 memcpy( &info->ProviderId, &ProviderIdSPX, sizeof(GUID) );
151 info->dwCatalogEntryId = 0x407;
152 info->ProtocolChain.ChainLen = 1;
153 info->iVersion = 2;
154 info->iAddressFamily = WS_AF_IPX;
155 info->iMaxSockAddr = 0x10;
156 info->iMinSockAddr = 0x0e;
157 info->iSocketType = 5;
158 info->dwMessageSize = 0xffffffff;
159 strcpyW( info->szProtocol, NameSpxW );
160 break;
162 case NSPROTO_SPXII:
163 info->dwServiceFlags1 = XP1_IFS_HANDLES | XP1_GRACEFUL_CLOSE |
164 XP1_PSEUDO_STREAM | XP1_MESSAGE_ORIENTED |
165 XP1_GUARANTEED_ORDER | XP1_GUARANTEED_DELIVERY;
166 memcpy( &info->ProviderId, &ProviderIdSPX, sizeof(GUID) );
167 info->dwCatalogEntryId = 0x409;
168 info->ProtocolChain.ChainLen = 1;
169 info->iVersion = 2;
170 info->iAddressFamily = WS_AF_IPX;
171 info->iMaxSockAddr = 0x10;
172 info->iMinSockAddr = 0x0e;
173 info->iSocketType = 5;
174 info->dwMessageSize = 0xffffffff;
175 strcpyW( info->szProtocol, NameSpxIIW );
176 break;
178 default:
179 if ((protocol == ISOPROTO_TP4) || (protocol == NSPROTO_SPX))
180 FIXME("Protocol <%s> not implemented\n",
181 (protocol == ISOPROTO_TP4) ? "ISOPROTO_TP4" : "NSPROTO_SPX");
182 else
183 FIXME("unknown Protocol <0x%08x>\n", protocol);
184 return SOCKET_ERROR;
186 return 1;
189 /*****************************************************************************
190 * WINSOCK_EnterSingleProtocolA [internal]
192 * see function WINSOCK_EnterSingleProtocolW
195 static INT WINSOCK_EnterSingleProtocolA( INT protocol, WSAPROTOCOL_INFOA* info )
197 memset( info, 0, sizeof(WSAPROTOCOL_INFOA) );
198 info->iProtocol = protocol;
200 switch (protocol)
202 case WS_IPPROTO_TCP:
203 info->dwServiceFlags1 = XP1_IFS_HANDLES | XP1_EXPEDITED_DATA |
204 XP1_GRACEFUL_CLOSE | XP1_GUARANTEED_ORDER |
205 XP1_GUARANTEED_DELIVERY;
206 memcpy( &info->ProviderId, &ProviderIdIP, sizeof(GUID) );
207 info->dwCatalogEntryId = 0x3e9;
208 info->ProtocolChain.ChainLen = 1;
209 info->iVersion = 2;
210 info->iAddressFamily = WS_AF_INET;
211 info->iMaxSockAddr = 0x10;
212 info->iMinSockAddr = 0x10;
213 info->iSocketType = WS_SOCK_STREAM;
214 strcpy( info->szProtocol, NameTcp );
215 break;
217 case WS_IPPROTO_UDP:
218 info->dwServiceFlags1 = XP1_IFS_HANDLES | XP1_SUPPORT_BROADCAST |
219 XP1_SUPPORT_MULTIPOINT | XP1_MESSAGE_ORIENTED |
220 XP1_CONNECTIONLESS;
221 memcpy( &info->ProviderId, &ProviderIdIP, sizeof(GUID) );
222 info->dwCatalogEntryId = 0x3ea;
223 info->ProtocolChain.ChainLen = 1;
224 info->iVersion = 2;
225 info->iAddressFamily = WS_AF_INET;
226 info->iMaxSockAddr = 0x10;
227 info->iMinSockAddr = 0x10;
228 info->iSocketType = WS_SOCK_DGRAM;
229 info->dwMessageSize = 0xffbb;
230 strcpy( info->szProtocol, NameUdp );
231 break;
233 case NSPROTO_IPX:
234 info->dwServiceFlags1 = XP1_IFS_HANDLES | XP1_SUPPORT_BROADCAST |
235 XP1_SUPPORT_MULTIPOINT | XP1_MESSAGE_ORIENTED |
236 XP1_CONNECTIONLESS;
237 memcpy( &info->ProviderId, &ProviderIdIPX, sizeof(GUID) );
238 info->dwCatalogEntryId = 0x406;
239 info->ProtocolChain.ChainLen = 1;
240 info->iVersion = 2;
241 info->iAddressFamily = WS_AF_IPX;
242 info->iMaxSockAddr = 0x10;
243 info->iMinSockAddr = 0x0e;
244 info->iSocketType = WS_SOCK_DGRAM;
245 info->iProtocolMaxOffset = 0xff;
246 info->dwMessageSize = 0x240;
247 strcpy( info->szProtocol, NameIpx );
248 break;
250 case NSPROTO_SPX:
251 info->dwServiceFlags1 = XP1_IFS_HANDLES | XP1_PSEUDO_STREAM |
252 XP1_MESSAGE_ORIENTED | XP1_GUARANTEED_ORDER |
253 XP1_GUARANTEED_DELIVERY;
254 memcpy( &info->ProviderId, &ProviderIdSPX, sizeof(GUID) );
255 info->dwCatalogEntryId = 0x407;
256 info->ProtocolChain.ChainLen = 1;
257 info->iVersion = 2;
258 info->iAddressFamily = WS_AF_IPX;
259 info->iMaxSockAddr = 0x10;
260 info->iMinSockAddr = 0x0e;
261 info->iSocketType = 5;
262 info->dwMessageSize = 0xffffffff;
263 strcpy( info->szProtocol, NameSpx );
264 break;
266 case NSPROTO_SPXII:
267 info->dwServiceFlags1 = XP1_IFS_HANDLES | XP1_GRACEFUL_CLOSE |
268 XP1_PSEUDO_STREAM | XP1_MESSAGE_ORIENTED |
269 XP1_GUARANTEED_ORDER | XP1_GUARANTEED_DELIVERY;
270 memcpy( &info->ProviderId, &ProviderIdSPX, sizeof(GUID) );
271 info->dwCatalogEntryId = 0x409;
272 info->ProtocolChain.ChainLen = 1;
273 info->iVersion = 2;
274 info->iAddressFamily = WS_AF_IPX;
275 info->iMaxSockAddr = 0x10;
276 info->iMinSockAddr = 0x0e;
277 info->iSocketType = 5;
278 info->dwMessageSize = 0xffffffff;
279 strcpy( info->szProtocol, NameSpxII );
280 break;
282 default:
283 if ((protocol == ISOPROTO_TP4) || (protocol == NSPROTO_SPX))
284 FIXME("Protocol <%s> not implemented\n",
285 (protocol == ISOPROTO_TP4) ? "ISOPROTO_TP4" : "NSPROTO_SPX");
286 else
287 FIXME("unknown Protocol <0x%08x>\n", protocol);
288 return SOCKET_ERROR;
290 return 1;
293 /*****************************************************************************
294 * WSAEnumProtocolsA [WS2_32.@]
296 * see function WSAEnumProtocolsW
298 INT WINAPI WSAEnumProtocolsA( LPINT protocols, LPWSAPROTOCOL_INFOA buffer, LPDWORD len )
300 INT i = 0;
301 DWORD size = 0;
302 INT local[] = { WS_IPPROTO_TCP, WS_IPPROTO_UDP, NSPROTO_IPX, NSPROTO_SPX, NSPROTO_SPXII, 0 };
304 if (!buffer)
305 return SOCKET_ERROR;
307 if (!protocols) protocols = local;
309 while (protocols[i]) i++;
311 size = i * sizeof(WSAPROTOCOL_INFOA);
313 if (*len < size)
315 *len = size;
316 return SOCKET_ERROR;
319 for (i = 0; protocols[i]; i++)
321 if (WINSOCK_EnterSingleProtocolA( protocols[i], &buffer[i] ) == SOCKET_ERROR)
322 return i;
324 return i;
327 /*****************************************************************************
328 * WSAEnumProtocolsW [WS2_32.@]
330 * Retrieves information about specified set of active network protocols.
332 * PARAMS
333 * protocols [I] Pointer to null-terminated array of protocol id's. NULL
334 * retrieves information on all available protocols.
335 * buffer [I] Pointer to a buffer to be filled with WSAPROTOCOL_INFO
336 * structures.
337 * len [I/O] Pointer to a variable specifying buffer size. On output
338 * the variable holds the number of bytes needed when the
339 * specified size is too small.
341 * RETURNS
342 * Success: number of WSAPROTOCOL_INFO structures in buffer.
343 * Failure: SOCKET_ERROR
345 * NOTES
346 * NT4SP5 does not return SPX if protocols == NULL
348 * BUGS
349 * - NT4SP5 returns in addition these list of NETBIOS protocols
350 * (address family 17), each entry two times one for socket type 2 and 5
352 * iProtocol szProtocol
353 * 0x80000000 \Device\NwlnkNb
354 * 0xfffffffa \Device\NetBT_CBENT7
355 * 0xfffffffb \Device\Nbf_CBENT7
356 * 0xfffffffc \Device\NetBT_NdisWan5
357 * 0xfffffffd \Device\NetBT_El9202
358 * 0xfffffffe \Device\Nbf_El9202
359 * 0xffffffff \Device\Nbf_NdisWan4
361 * - there is no check that the operating system supports the returned
362 * protocols
364 INT WINAPI WSAEnumProtocolsW( LPINT protocols, LPWSAPROTOCOL_INFOW buffer, LPDWORD len )
366 INT i = 0;
367 DWORD size = 0;
368 INT local[] = { WS_IPPROTO_TCP, WS_IPPROTO_UDP, NSPROTO_IPX, NSPROTO_SPX, NSPROTO_SPXII, 0 };
370 if (!buffer)
371 return SOCKET_ERROR;
373 if (!protocols) protocols = local;
375 while (protocols[i]) i++;
377 size = i * sizeof(WSAPROTOCOL_INFOW);
379 if (*len < size)
381 *len = size;
382 return SOCKET_ERROR;
385 for (i = 0; protocols[i]; i++)
387 if (WINSOCK_EnterSingleProtocolW( protocols[i], &buffer[i] ) == SOCKET_ERROR)
388 return i;
390 return i;
393 /*****************************************************************************
394 * WSCEnumProtocols [WS2_32.@]
396 * PARAMS
397 * protocols [I] Null-terminated array of iProtocol values.
398 * buffer [O] Buffer of WSAPROTOCOL_INFOW structures.
399 * len [I/O] Size of buffer on input/output.
400 * errno [O] Error code.
402 * RETURNS
403 * Success: number of protocols to be reported on.
404 * Failure: SOCKET_ERROR. error is in errno.
406 * BUGS
407 * Doesn't supply info on layered protocols.
410 INT WINAPI WSCEnumProtocols( LPINT protocols, LPWSAPROTOCOL_INFOW buffer, LPDWORD len, LPINT errno )
412 INT ret = WSAEnumProtocolsW( protocols, buffer, len );
414 if (ret == SOCKET_ERROR) *errno = WSAENOBUFS;
416 return ret;