1 /* Copyright (c) 2003 Juan Lang
3 * This library is free software; you can redistribute it and/or
4 * modify it under the terms of the GNU Lesser General Public
5 * License as published by the Free Software Foundation; either
6 * version 2.1 of the License, or (at your option) any later version.
8 * This library is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * Lesser General Public License for more details.
13 * You should have received a copy of the GNU Lesser General Public
14 * License along with this library; if not, write to the Free Software
15 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17 * I am heavily indebted to Chris Hertel's excellent Implementing CIFS,
18 * http://ubiqx.org/cifs/ , for whatever understanding I have of NBT.
19 * I also stole from Mike McCormack's smb.c and netapi32.c, although little of
21 * Lack of understanding and bugs are my fault.
24 * - Of the NetBIOS session functions, only client functions are supported, and
25 * it's likely they'll be the only functions supported. NBT requires session
26 * servers to listen on TCP/139. This requires root privilege, and Samba is
27 * likely to be listening here already. This further restricts NetBIOS
28 * applications, both explicit users and implicit ones: CreateNamedPipe
29 * won't actually create a listening pipe, for example, so applications can't
30 * act as RPC servers using a named pipe protocol binding, DCOM won't be able
31 * to support callbacks or servers over the named pipe protocol, etc.
33 * - Datagram support is omitted for the same reason. To send a NetBIOS
34 * datagram, you must include the NetBIOS name by which your application is
35 * known. This requires you to have registered the name previously, and be
36 * able to act as a NetBIOS datagram server (listening on UDP/138).
38 * - Name registration functions are omitted for the same reason--registering a
39 * name requires you to be able to defend it, and this means listening on
41 * Win98 requires you either use your computer's NetBIOS name (with the NULL
42 * suffix byte) as the calling name when creating a session, or to register
43 * a new name before creating one: it disallows '*' as the calling name.
44 * Win2K initially starts with an empty name table, and doesn't allow you to
45 * use the machine's NetBIOS name (with the NULL suffix byte) as the calling
46 * name. Although it allows sessions to be created with '*' as the calling
47 * name, doing so results in timeouts for all receives, because the
48 * application never gets them.
49 * So, a well-behaved NetBIOS application will typically want to register a
50 * name. I should probably support a do-nothing name list that allows
51 * NCBADDNAME to add to it, but doesn't actually register the name, or does
52 * attempt to register it without being able to defend it.
54 * - Name lookups may not behave quite as you'd expect/like if you have
55 * multiple LANAs. If a name is resolvable through DNS, or if you're using
56 * WINS, it'll resolve on _any_ LANA. So, a Call will succeed on any LANA as
58 * I'm not sure how Windows behaves in this case. I could try to force
59 * lookups to the correct adapter by using one of the GetPreferred*
60 * functions, but with the possibility of multiple adapters in the same
61 * same subnet, there's no guarantee that what IpHlpApi thinks is the
62 * preferred adapter will actually be a LANA. (It's highly probable because
63 * this is an unusual configuration, but not guaranteed.)
65 * See also other FIXMEs in the code.
74 #include "wine/debug.h"
79 #include "nbnamecache.h"
81 WINE_DEFAULT_DEBUG_CHANNEL(netbios);
88 #define INADDR_NONE ~0UL
91 #define NBR_ADDWORD(p,word) (*(WORD *)(p)) = htons(word)
92 #define NBR_GETWORD(p) ntohs(*(WORD *)(p))
95 #define MAX_QUERIES 0xffff
96 #define MIN_QUERY_TIMEOUT 100
97 #define MAX_QUERY_TIMEOUT 0xffffffff
98 #define BCAST_QUERIES 3
99 #define BCAST_QUERY_TIMEOUT 750
100 #define WINS_QUERIES 3
101 #define WINS_QUERY_TIMEOUT 750
102 #define MAX_WINS_SERVERS 2
103 #define MIN_CACHE_TIMEOUT 60000
104 #define CACHE_TIMEOUT 360000
106 #define MAX_NBT_NAME_SZ (NCBNAMSZ * 2 + MAX_DOMAIN_NAME_LEN + 2)
107 #define SIMPLE_NAME_QUERY_PKT_SIZE 26 + MAX_NBT_NAME_SZ
109 #define DEFAULT_NBT_SESSIONS 16
111 #define NBNS_TYPE_NB 0x0020
112 #define NBNS_TYPE_NBSTAT 0x0021
113 #define NBNS_CLASS_INTERNET 0x00001
114 #define NBNS_HEADER_SIZE (sizeof(WORD) * 6)
115 #define NBNS_RESPONSE_AND_OPCODE 0xf800
116 #define NBNS_RESPONSE_AND_QUERY 0x8000
117 #define NBNS_REPLYCODE 0x0f
119 #define NBSS_HDRSIZE 4
121 #define NBSS_MSG 0x00
122 #define NBSS_REQ 0x81
123 #define NBSS_ACK 0x82
124 #define NBSS_NACK 0x83
125 #define NBSS_RETARGET 0x84
126 #define NBSS_KEEPALIVE 0x85
128 #define NBSS_ERR_NOT_LISTENING_ON_NAME 0x80
129 #define NBSS_ERR_NOT_LISTENING_FOR_CALLER 0x81
130 #define NBSS_ERR_BAD_NAME 0x82
131 #define NBSS_ERR_INSUFFICIENT_RESOURCES 0x83
133 #define NBSS_EXTENSION 0x01
135 typedef struct _NetBTSession
142 typedef struct _NetBTAdapter
146 struct NBNameCache *nameCache;
151 static ULONG gTransportID;
152 static BOOL gEnableDNS;
153 static DWORD gBCastQueries;
154 static DWORD gBCastQueryTimeout;
155 static DWORD gWINSQueries;
156 static DWORD gWINSQueryTimeout;
157 static DWORD gWINSServers[MAX_WINS_SERVERS];
158 static int gNumWINSServers;
159 static char gScopeID[MAX_DOMAIN_NAME_LEN];
160 static DWORD gCacheTimeout;
161 static struct NBNameCache *gNameCache;
163 /* Converts from a NetBIOS name into a Second Level Encoding-formatted name.
164 * Assumes p is not NULL and is either NULL terminated or has at most NCBNAMSZ
165 * bytes, and buffer has at least MAX_NBT_NAME_SZ bytes. Pads with space bytes
166 * if p is NULL-terminated. Returns the number of bytes stored in buffer.
168 static int NetBTNameEncode(const UCHAR *p, UCHAR *buffer)
173 if (!buffer) return 0;
175 buffer[len++] = NCBNAMSZ * 2;
176 for (i = 0; p[i] && i < NCBNAMSZ; i++)
178 buffer[len++] = ((p[i] & 0xf0) >> 4) + 'A';
179 buffer[len++] = (p[i] & 0x0f) + 'A';
181 while (len < NCBNAMSZ * 2)
188 int scopeIDLen = strlen(gScopeID);
190 memcpy(buffer + len, gScopeID, scopeIDLen);
193 buffer[len++] = 0; /* add second terminator */
197 /* Creates a NBT name request packet for name in buffer. If broadcast is true,
198 * creates a broadcast request, otherwise creates a unicast request.
199 * Returns the number of bytes stored in buffer.
201 static DWORD NetBTNameReq(const UCHAR name[NCBNAMSZ], WORD xid, WORD qtype,
202 BOOL broadcast, UCHAR *buffer, int len)
206 if (len < SIMPLE_NAME_QUERY_PKT_SIZE) return 0;
208 NBR_ADDWORD(&buffer[i],xid); i+=2; /* transaction */
211 NBR_ADDWORD(&buffer[i],0x0110); /* flags: r=req,op=query,rd=1,b=1 */
216 NBR_ADDWORD(&buffer[i],0x0100); /* flags: r=req,op=query,rd=1,b=0 */
219 NBR_ADDWORD(&buffer[i],0x0001); i+=2; /* one name query */
220 NBR_ADDWORD(&buffer[i],0x0000); i+=2; /* zero answers */
221 NBR_ADDWORD(&buffer[i],0x0000); i+=2; /* zero authorities */
222 NBR_ADDWORD(&buffer[i],0x0000); i+=2; /* zero additional */
224 i += NetBTNameEncode(name, &buffer[i]);
226 NBR_ADDWORD(&buffer[i],qtype); i+=2;
227 NBR_ADDWORD(&buffer[i],NBNS_CLASS_INTERNET); i+=2;
232 /* Sends a name query request for name on fd to destAddr. Sets SO_BROADCAST on
233 * fd if broadcast is TRUE. Assumes fd is not INVALID_SOCKET, and name is not
235 * Returns 0 on success, -1 on failure.
237 static int NetBTSendNameQuery(SOCKET fd, const UCHAR name[NCBNAMSZ], WORD xid,
238 WORD qtype, DWORD destAddr, BOOL broadcast)
243 addr.s_addr = destAddr;
244 TRACE("name %s, dest addr %s\n", name, inet_ntoa(addr));
247 ret = setsockopt(fd, SOL_SOCKET, SO_BROADCAST, (const char*)&on, sizeof(on));
251 UCHAR buf[SIMPLE_NAME_QUERY_PKT_SIZE];
252 struct sockaddr_in sin;
254 memset(&sin, 0, sizeof(sin));
255 sin.sin_addr.s_addr = destAddr;
256 sin.sin_family = AF_INET;
257 sin.sin_port = htons(PORT_NBNS);
259 wsaBuf.buf = (CHAR*)buf;
260 wsaBuf.len = NetBTNameReq(name, xid, qtype, broadcast, buf,
266 ret = WSASendTo(fd, &wsaBuf, 1, &bytesSent, 0,
267 (struct sockaddr*)&sin, sizeof(sin), NULL, NULL);
268 if (ret < 0 || bytesSent < wsaBuf.len)
279 typedef BOOL (*NetBTAnswerCallback)(void *data, WORD answerCount,
280 WORD answerIndex, PUCHAR rData, WORD rdLength);
282 /* Waits on fd until GetTickCount() returns a value greater than or equal to
283 * waitUntil for a name service response. If a name response matching xid
284 * is received, calls answerCallback once for each answer resource record in
285 * the response. (The callback's answerCount will be the total number of
286 * answers to expect, and answerIndex will be the 0-based index that's being
287 * sent this time.) Quits parsing if answerCallback returns FALSE.
288 * Returns NRC_GOODRET on timeout or a valid response received, something else
291 static UCHAR NetBTWaitForNameResponse(const NetBTAdapter *adapter, SOCKET fd,
292 DWORD waitUntil, NetBTAnswerCallback answerCallback, void *data)
296 UCHAR ret = NRC_GOODRET;
298 if (!adapter) return NRC_BADDR;
299 if (fd == INVALID_SOCKET) return NRC_BADDR;
300 if (!answerCallback) return NRC_BADDR;
302 while (!found && ret == NRC_GOODRET && (now = GetTickCount()) < waitUntil)
304 DWORD msToWait = waitUntil - now;
306 struct timeval timeout = { msToWait / 1000, msToWait % 1000 };
311 r = select(fd + 1, &fds, NULL, NULL, &timeout);
316 /* FIXME: magic #, is this always enough? */
319 struct sockaddr_in fromaddr;
320 WORD respXID, flags, queryCount, answerCount;
321 WSABUF wsaBuf = { sizeof(buffer), (CHAR*)buffer };
322 DWORD bytesReceived, recvFlags = 0;
324 fromsize = sizeof(fromaddr);
325 r = WSARecvFrom(fd, &wsaBuf, 1, &bytesReceived, &recvFlags,
326 (struct sockaddr*)&fromaddr, &fromsize, NULL, NULL);
333 if (bytesReceived < NBNS_HEADER_SIZE)
336 respXID = NBR_GETWORD(buffer);
337 if (adapter->nameQueryXID != respXID)
340 flags = NBR_GETWORD(buffer + 2);
341 queryCount = NBR_GETWORD(buffer + 4);
342 answerCount = NBR_GETWORD(buffer + 6);
344 /* a reply shouldn't contain a query, ignore bad packet */
348 if ((flags & NBNS_RESPONSE_AND_OPCODE) == NBNS_RESPONSE_AND_QUERY)
350 if ((flags & NBNS_REPLYCODE) != 0)
352 else if ((flags & NBNS_REPLYCODE) == 0 && answerCount > 0)
354 PUCHAR ptr = buffer + NBNS_HEADER_SIZE;
355 BOOL shouldContinue = TRUE;
356 WORD answerIndex = 0;
359 /* decode one answer at a time */
360 while (ret == NRC_GOODRET && answerIndex < answerCount &&
361 ptr - buffer < bytesReceived && shouldContinue)
366 for (; ptr[0] && ptr - buffer < bytesReceived; )
369 ptr += 2; /* scan past type */
370 if (ptr - buffer < bytesReceived && ret == NRC_GOODRET
371 && NBR_GETWORD(ptr) == NBNS_CLASS_INTERNET)
374 ret = NRC_SYSTEM; /* parse error */
375 ptr += sizeof(DWORD); /* TTL */
376 rLen = NBR_GETWORD(ptr);
377 rLen = min(rLen, bytesReceived - (ptr - buffer));
379 shouldContinue = answerCallback(data, answerCount,
380 answerIndex, ptr, rLen);
388 TRACE("returning 0x%02x\n", ret);
392 typedef struct _NetBTNameQueryData {
393 NBNameCacheEntry *cacheEntry;
395 } NetBTNameQueryData;
397 /* Name query callback function for NetBTWaitForNameResponse, creates a cache
398 * entry on the first answer, adds each address as it's called again (as long
399 * as there's space). If there's an error that should be propagated as the
400 * NetBIOS error, modifies queryData's ret member to the proper return code.
402 static BOOL NetBTFindNameAnswerCallback(void *pVoid, WORD answerCount,
403 WORD answerIndex, PUCHAR rData, WORD rLen)
405 NetBTNameQueryData *queryData = (NetBTNameQueryData *)pVoid;
410 if (queryData->cacheEntry == NULL)
412 queryData->cacheEntry = HeapAlloc(
413 GetProcessHeap(), 0, sizeof(NBNameCacheEntry) +
414 (answerCount - 1) * sizeof(DWORD));
415 if (queryData->cacheEntry)
416 queryData->cacheEntry->numAddresses = 0;
418 queryData->ret = NRC_OSRESNOTAV;
420 if (rLen == 6 && queryData->cacheEntry &&
421 queryData->cacheEntry->numAddresses < answerCount)
423 queryData->cacheEntry->addresses[queryData->cacheEntry->
424 numAddresses++] = *(const DWORD *)(rData + 2);
425 ret = queryData->cacheEntry->numAddresses < answerCount;
435 /* Workhorse NetBT name lookup function. Sends a name lookup query for
436 * ncb->ncb_callname to sendTo, as a broadcast if broadcast is TRUE, using
437 * adapter->nameQueryXID as the transaction ID. Waits up to timeout
438 * milliseconds, and retries up to maxQueries times, waiting for a reply.
439 * If a valid response is received, stores the looked up addresses as a
440 * NBNameCacheEntry in *cacheEntry.
441 * Returns NRC_GOODRET on success, though this may not mean the name was
442 * resolved--check whether *cacheEntry is NULL.
444 static UCHAR NetBTNameWaitLoop(const NetBTAdapter *adapter, SOCKET fd, const NCB *ncb,
445 DWORD sendTo, BOOL broadcast, DWORD timeout, DWORD maxQueries,
446 NBNameCacheEntry **cacheEntry)
448 unsigned int queries;
449 NetBTNameQueryData queryData;
451 if (!adapter) return NRC_BADDR;
452 if (fd == INVALID_SOCKET) return NRC_BADDR;
453 if (!ncb) return NRC_BADDR;
454 if (!cacheEntry) return NRC_BADDR;
456 queryData.cacheEntry = NULL;
457 queryData.ret = NRC_GOODRET;
458 for (queries = 0; queryData.cacheEntry == NULL && queries < maxQueries;
461 if (!NCB_CANCELLED(ncb))
463 int r = NetBTSendNameQuery(fd, ncb->ncb_callname,
464 adapter->nameQueryXID, NBNS_TYPE_NB, sendTo, broadcast);
467 queryData.ret = NetBTWaitForNameResponse(adapter, fd,
468 GetTickCount() + timeout, NetBTFindNameAnswerCallback,
471 queryData.ret = NRC_SYSTEM;
474 queryData.ret = NRC_CMDCAN;
476 if (queryData.cacheEntry)
478 memcpy(queryData.cacheEntry->name, ncb->ncb_callname, NCBNAMSZ);
479 memcpy(queryData.cacheEntry->nbname, ncb->ncb_callname, NCBNAMSZ);
481 *cacheEntry = queryData.cacheEntry;
482 return queryData.ret;
485 /* Attempts to add cacheEntry to the name cache in *nameCache; if *nameCache
486 * has not yet been created, creates it, using gCacheTimeout as the cache
487 * entry timeout. If memory allocation fails, or if NBNameCacheAddEntry fails,
489 * Returns NRC_GOODRET on success, and something else on failure.
491 static UCHAR NetBTStoreCacheEntry(struct NBNameCache **nameCache,
492 NBNameCacheEntry *cacheEntry)
496 if (!nameCache) return NRC_BADDR;
497 if (!cacheEntry) return NRC_BADDR;
500 *nameCache = NBNameCacheCreate(GetProcessHeap(), gCacheTimeout);
502 ret = NBNameCacheAddEntry(*nameCache, cacheEntry)
503 ? NRC_GOODRET : NRC_OSRESNOTAV;
506 HeapFree(GetProcessHeap(), 0, cacheEntry);
507 ret = NRC_OSRESNOTAV;
512 /* Attempts to resolve name using inet_addr(), then gethostbyname() if
513 * gEnableDNS is TRUE, if the suffix byte is either <00> or <20>. If the name
514 * can be looked up, returns 0 and stores the looked up addresses as a
515 * NBNameCacheEntry in *cacheEntry.
516 * Returns NRC_GOODRET on success, though this may not mean the name was
517 * resolved--check whether *cacheEntry is NULL. Returns something else on
520 static UCHAR NetBTinetResolve(const UCHAR name[NCBNAMSZ],
521 NBNameCacheEntry **cacheEntry)
523 UCHAR ret = NRC_GOODRET;
525 TRACE("name %s, cacheEntry %p\n", name, cacheEntry);
527 if (!name) return NRC_BADDR;
528 if (!cacheEntry) return NRC_BADDR;
530 if (isalnum(name[0]) && (name[NCBNAMSZ - 1] == 0 ||
531 name[NCBNAMSZ - 1] == 0x20))
533 CHAR toLookup[NCBNAMSZ];
536 for (i = 0; i < NCBNAMSZ - 1 && name[i] && name[i] != ' '; i++)
537 toLookup[i] = name[i];
540 if (isdigit(toLookup[0]))
542 unsigned long addr = inet_addr(toLookup);
544 if (addr != INADDR_NONE)
546 *cacheEntry = HeapAlloc(GetProcessHeap(),
547 0, sizeof(NBNameCacheEntry));
550 memcpy((*cacheEntry)->name, name, NCBNAMSZ);
551 memset((*cacheEntry)->nbname, 0, NCBNAMSZ);
552 (*cacheEntry)->nbname[0] = '*';
553 (*cacheEntry)->numAddresses = 1;
554 (*cacheEntry)->addresses[0] = addr;
557 ret = NRC_OSRESNOTAV;
560 if (gEnableDNS && ret == NRC_GOODRET && !*cacheEntry)
562 struct hostent *host;
564 if ((host = gethostbyname(toLookup)) != NULL)
566 for (i = 0; ret == NRC_GOODRET && host->h_addr_list &&
567 host->h_addr_list[i]; i++)
569 if (host->h_addr_list && host->h_addr_list[0])
571 *cacheEntry = HeapAlloc(
572 GetProcessHeap(), 0, sizeof(NBNameCacheEntry) +
573 (i - 1) * sizeof(DWORD));
576 memcpy((*cacheEntry)->name, name, NCBNAMSZ);
577 memset((*cacheEntry)->nbname, 0, NCBNAMSZ);
578 (*cacheEntry)->nbname[0] = '*';
579 (*cacheEntry)->numAddresses = i;
580 for (i = 0; i < (*cacheEntry)->numAddresses; i++)
581 (*cacheEntry)->addresses[i] =
582 (DWORD)host->h_addr_list[i];
585 ret = NRC_OSRESNOTAV;
591 TRACE("returning 0x%02x\n", ret);
595 /* Looks up the name in ncb->ncb_callname, first in the name caches (global
596 * and this adapter's), then using gethostbyname(), next by WINS if configured,
597 * and finally using broadcast NetBT name resolution. In NBT parlance, this
598 * makes this an "H-node". Stores an entry in the appropriate name cache for a
599 * found node, and returns it as *cacheEntry.
600 * Assumes data, ncb, and cacheEntry are not NULL.
601 * Returns NRC_GOODRET on success--which doesn't mean the name was resolved,
602 * just that all name lookup operations completed successfully--and something
603 * else on failure. *cacheEntry will be NULL if the name was not found.
605 static UCHAR NetBTInternalFindName(NetBTAdapter *adapter, PNCB ncb,
606 const NBNameCacheEntry **cacheEntry)
608 UCHAR ret = NRC_GOODRET;
610 TRACE("adapter %p, ncb %p, cacheEntry %p\n", adapter, ncb, cacheEntry);
612 if (!cacheEntry) return NRC_BADDR;
615 if (!adapter) return NRC_BADDR;
616 if (!ncb) return NRC_BADDR;
618 if (ncb->ncb_callname[0] == '*')
622 *cacheEntry = NBNameCacheFindEntry(gNameCache, ncb->ncb_callname);
624 *cacheEntry = NBNameCacheFindEntry(adapter->nameCache,
628 NBNameCacheEntry *newEntry = NULL;
630 ret = NetBTinetResolve(ncb->ncb_callname, &newEntry);
631 if (ret == NRC_GOODRET && newEntry)
633 ret = NetBTStoreCacheEntry(&gNameCache, newEntry);
634 if (ret != NRC_GOODRET)
639 SOCKET fd = WSASocketA(PF_INET, SOCK_DGRAM, IPPROTO_UDP, NULL,
640 0, WSA_FLAG_OVERLAPPED);
642 if(fd == INVALID_SOCKET)
643 ret = NRC_OSRESNOTAV;
648 adapter->nameQueryXID++;
649 for (winsNdx = 0; ret == NRC_GOODRET && *cacheEntry == NULL
650 && winsNdx < gNumWINSServers; winsNdx++)
651 ret = NetBTNameWaitLoop(adapter, fd, ncb,
652 gWINSServers[winsNdx], FALSE, gWINSQueryTimeout,
653 gWINSQueries, &newEntry);
654 if (ret == NRC_GOODRET && newEntry)
656 ret = NetBTStoreCacheEntry(&gNameCache, newEntry);
657 if (ret != NRC_GOODRET)
660 if (ret == NRC_GOODRET && *cacheEntry == NULL)
663 adapter->ipr.dwAddr & adapter->ipr.dwMask;
665 if (adapter->ipr.dwBCastAddr)
666 bcastAddr |= ~adapter->ipr.dwMask;
667 ret = NetBTNameWaitLoop(adapter, fd, ncb, bcastAddr,
668 TRUE, gBCastQueryTimeout, gBCastQueries, &newEntry);
669 if (ret == NRC_GOODRET && newEntry)
671 ret = NetBTStoreCacheEntry(&adapter->nameCache,
673 if (ret != NRC_GOODRET)
680 *cacheEntry = newEntry;
683 TRACE("returning 0x%02x\n", ret);
687 typedef struct _NetBTNodeQueryData
690 PADAPTER_STATUS astat;
692 } NetBTNodeQueryData;
694 /* Callback function for NetBTAstatRemote, parses the rData for the node
695 * status and name list of the remote node. Always returns FALSE, since
696 * there's never more than one answer we care about in a node status response.
698 static BOOL NetBTNodeStatusAnswerCallback(void *pVoid, WORD answerCount,
699 WORD answerIndex, PUCHAR rData, WORD rLen)
701 NetBTNodeQueryData *data = (NetBTNodeQueryData *)pVoid;
703 if (data && !data->gotResponse && rData && rLen >= 1)
705 /* num names is first byte; each name is NCBNAMSZ + 2 bytes */
706 if (rLen >= rData[0] * (NCBNAMSZ + 2))
712 data->gotResponse = TRUE;
713 data->astat->name_count = rData[0];
714 for (i = 0, src = rData + 1,
715 dst = (PNAME_BUFFER)((PUCHAR)data->astat +
716 sizeof(ADAPTER_STATUS));
717 i < data->astat->name_count && src - rData < rLen &&
718 (PUCHAR)dst - (PUCHAR)data->astat < data->astatLen;
719 i++, dst++, src += NCBNAMSZ + 2)
721 UCHAR flags = *(src + NCBNAMSZ);
723 memcpy(dst->name, src, NCBNAMSZ);
724 /* we won't actually see a registering name in the returned
725 * response. It's useful to see if no other flags are set; if
726 * none are, then the name is registered. */
727 dst->name_flags = REGISTERING;
729 dst->name_flags |= GROUP_NAME;
731 dst->name_flags |= DEREGISTERED;
733 dst->name_flags |= DUPLICATE;
734 if (dst->name_flags == REGISTERING)
735 dst->name_flags = REGISTERED;
737 /* arbitrarily set HW type to Ethernet */
738 data->astat->adapter_type = 0xfe;
739 if (src - rData < rLen)
740 memcpy(data->astat->adapter_address, src,
741 min(rLen - (src - rData), 6));
747 /* This uses the WINS timeout and query values, as they're the
748 * UCAST_REQ_RETRY_TIMEOUT and UCAST_REQ_RETRY_COUNT according to the RFCs.
750 static UCHAR NetBTAstatRemote(NetBTAdapter *adapter, PNCB ncb)
752 UCHAR ret = NRC_GOODRET;
753 const NBNameCacheEntry *cacheEntry = NULL;
755 TRACE("adapter %p, NCB %p\n", adapter, ncb);
757 if (!adapter) return NRC_BADDR;
758 if (!ncb) return NRC_INVADDRESS;
760 ret = NetBTInternalFindName(adapter, ncb, &cacheEntry);
761 if (ret == NRC_GOODRET && cacheEntry)
763 if (cacheEntry->numAddresses > 0)
765 SOCKET fd = WSASocketA(PF_INET, SOCK_DGRAM, IPPROTO_UDP, NULL, 0,
766 WSA_FLAG_OVERLAPPED);
768 if(fd == INVALID_SOCKET)
769 ret = NRC_OSRESNOTAV;
772 NetBTNodeQueryData queryData;
774 PADAPTER_STATUS astat = (PADAPTER_STATUS)ncb->ncb_buffer;
776 adapter->nameQueryXID++;
777 astat->name_count = 0;
778 queryData.gotResponse = FALSE;
779 queryData.astat = astat;
780 queryData.astatLen = ncb->ncb_length;
781 for (queries = 0; !queryData.gotResponse &&
782 queries < gWINSQueries; queries++)
784 if (!NCB_CANCELLED(ncb))
786 int r = NetBTSendNameQuery(fd, ncb->ncb_callname,
787 adapter->nameQueryXID, NBNS_TYPE_NBSTAT,
788 cacheEntry->addresses[0], FALSE);
791 ret = NetBTWaitForNameResponse(adapter, fd,
792 GetTickCount() + gWINSQueryTimeout,
793 NetBTNodeStatusAnswerCallback, &queryData);
806 else if (ret == NRC_CMDCAN)
807 ; /* do nothing, we were cancelled */
810 TRACE("returning 0x%02x\n", ret);
814 static UCHAR NetBTAstat(void *adapt, PNCB ncb)
816 NetBTAdapter *adapter = (NetBTAdapter *)adapt;
819 TRACE("adapt %p, NCB %p\n", adapt, ncb);
821 if (!adapter) return NRC_ENVNOTDEF;
822 if (!ncb) return NRC_INVADDRESS;
823 if (!ncb->ncb_buffer) return NRC_BADDR;
824 if (ncb->ncb_length < sizeof(ADAPTER_STATUS)) return NRC_BUFLEN;
826 if (ncb->ncb_callname[0] == '*')
830 PADAPTER_STATUS astat = (PADAPTER_STATUS)ncb->ncb_buffer;
832 memset(astat, 0, sizeof(ADAPTER_STATUS));
833 astat->rev_major = 3;
834 ifRow.dwIndex = adapter->ipr.dwIndex;
835 if (GetIfEntry(&ifRow) != NO_ERROR)
839 physAddrLen = min(ifRow.dwPhysAddrLen, 6);
841 memcpy(astat->adapter_address, ifRow.bPhysAddr, physAddrLen);
842 /* doubt anyone cares, but why not.. */
843 if (ifRow.dwType == MIB_IF_TYPE_TOKENRING)
844 astat->adapter_type = 0xff;
846 astat->adapter_type = 0xfe; /* for Ethernet */
847 astat->max_sess_pkt_size = 0xffff;
848 astat->xmit_success = adapter->xmit_success;
849 astat->recv_success = adapter->recv_success;
854 ret = NetBTAstatRemote(adapter, ncb);
855 TRACE("returning 0x%02x\n", ret);
859 static UCHAR NetBTFindName(void *adapt, PNCB ncb)
861 NetBTAdapter *adapter = (NetBTAdapter *)adapt;
863 const NBNameCacheEntry *cacheEntry = NULL;
864 PFIND_NAME_HEADER foundName;
866 TRACE("adapt %p, NCB %p\n", adapt, ncb);
868 if (!adapter) return NRC_ENVNOTDEF;
869 if (!ncb) return NRC_INVADDRESS;
870 if (!ncb->ncb_buffer) return NRC_BADDR;
871 if (ncb->ncb_length < sizeof(FIND_NAME_HEADER)) return NRC_BUFLEN;
873 foundName = (PFIND_NAME_HEADER)ncb->ncb_buffer;
874 memset(foundName, 0, sizeof(FIND_NAME_HEADER));
876 ret = NetBTInternalFindName(adapter, ncb, &cacheEntry);
877 if (ret == NRC_GOODRET)
881 DWORD spaceFor = min((ncb->ncb_length - sizeof(FIND_NAME_HEADER)) /
882 sizeof(FIND_NAME_BUFFER), cacheEntry->numAddresses);
885 for (ndx = 0; ndx < spaceFor; ndx++)
887 PFIND_NAME_BUFFER findNameBuffer;
890 (PFIND_NAME_BUFFER)((PUCHAR)foundName +
891 sizeof(FIND_NAME_HEADER) + foundName->node_count *
892 sizeof(FIND_NAME_BUFFER));
893 memset(findNameBuffer->destination_addr, 0, 2);
894 memcpy(findNameBuffer->destination_addr + 2,
895 &adapter->ipr.dwAddr, sizeof(DWORD));
896 memset(findNameBuffer->source_addr, 0, 2);
897 memcpy(findNameBuffer->source_addr + 2,
898 &cacheEntry->addresses[ndx], sizeof(DWORD));
899 foundName->node_count++;
901 if (spaceFor < cacheEntry->numAddresses)
907 TRACE("returning 0x%02x\n", ret);
911 static UCHAR NetBTSessionReq(SOCKET fd, const UCHAR *calledName,
912 const UCHAR *callingName)
914 UCHAR buffer[NBSS_HDRSIZE + MAX_DOMAIN_NAME_LEN * 2], ret;
916 unsigned int len = 0;
917 DWORD bytesSent, bytesReceived, recvFlags = 0;
920 buffer[0] = NBSS_REQ;
923 len += NetBTNameEncode(calledName, &buffer[NBSS_HDRSIZE]);
924 len += NetBTNameEncode(callingName, &buffer[NBSS_HDRSIZE + len]);
926 NBR_ADDWORD(&buffer[2], len);
928 wsaBuf.len = len + NBSS_HDRSIZE;
929 wsaBuf.buf = (char*)buffer;
931 r = WSASend(fd, &wsaBuf, 1, &bytesSent, 0, NULL, NULL);
932 if(r < 0 || bytesSent < len + NBSS_HDRSIZE)
934 ERR("send failed\n");
938 /* I've already set the recv timeout on this socket (if it supports it), so
939 * just block. Hopefully we'll always receive the session acknowledgement
940 * within one timeout.
942 wsaBuf.len = NBSS_HDRSIZE + 1;
943 r = WSARecv(fd, &wsaBuf, 1, &bytesReceived, &recvFlags, NULL, NULL);
944 if (r < 0 || bytesReceived < NBSS_HDRSIZE)
946 else if (buffer[0] == NBSS_NACK)
948 if (r == NBSS_HDRSIZE + 1)
950 switch (buffer[NBSS_HDRSIZE])
952 case NBSS_ERR_INSUFFICIENT_RESOURCES:
962 else if (buffer[0] == NBSS_RETARGET)
964 FIXME("Got a session retarget, can't deal\n");
967 else if (buffer[0] == NBSS_ACK)
972 TRACE("returning 0x%02x\n", ret);
976 static UCHAR NetBTCall(void *adapt, PNCB ncb, void **sess)
978 NetBTAdapter *adapter = (NetBTAdapter *)adapt;
980 const NBNameCacheEntry *cacheEntry = NULL;
982 TRACE("adapt %p, ncb %p\n", adapt, ncb);
984 if (!adapter) return NRC_ENVNOTDEF;
985 if (!ncb) return NRC_INVADDRESS;
986 if (!sess) return NRC_BADDR;
988 ret = NetBTInternalFindName(adapter, ncb, &cacheEntry);
989 if (ret == NRC_GOODRET)
991 if (cacheEntry && cacheEntry->numAddresses > 0)
995 fd = WSASocketA(PF_INET, SOCK_STREAM, IPPROTO_TCP, NULL, 0,
996 WSA_FLAG_OVERLAPPED);
997 if (fd != INVALID_SOCKET)
1000 struct sockaddr_in sin;
1002 if (ncb->ncb_rto > 0)
1004 timeout = ncb->ncb_rto * 500;
1005 setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, (char*)&timeout,
1008 if (ncb->ncb_rto > 0)
1010 timeout = ncb->ncb_sto * 500;
1011 setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, (char*)&timeout,
1015 memset(&sin, 0, sizeof(sin));
1016 memcpy(&sin.sin_addr, &cacheEntry->addresses[0],
1017 sizeof(sin.sin_addr));
1018 sin.sin_family = AF_INET;
1019 sin.sin_port = htons(PORT_NBSS);
1020 /* FIXME: use nonblocking mode for the socket, check the
1021 * cancel flag periodically
1023 if (connect(fd, (struct sockaddr *)&sin, sizeof(sin))
1028 static const UCHAR fakedCalledName[] = "*SMBSERVER";
1029 const UCHAR *calledParty = cacheEntry->nbname[0] == '*'
1030 ? fakedCalledName : cacheEntry->nbname;
1032 ret = NetBTSessionReq(fd, calledParty, ncb->ncb_name);
1033 if (ret != NRC_GOODRET && calledParty[0] == '*')
1035 FIXME("NBT session to \"*SMBSERVER\" refused,\n");
1036 FIXME("should try finding name using ASTAT\n");
1039 if (ret != NRC_GOODRET)
1043 NetBTSession *session = HeapAlloc(
1044 GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(NetBTSession));
1049 InitializeCriticalSection(&session->cs);
1050 session->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": NetBTSession.cs");
1055 ret = NRC_OSRESNOTAV;
1061 ret = NRC_OSRESNOTAV;
1066 TRACE("returning 0x%02x\n", ret);
1070 /* Notice that I don't protect against multiple thread access to NetBTSend.
1071 * This is because I don't update any data in the adapter, and I only make a
1072 * single call to WSASend, which I assume to act atomically (not interleaving
1073 * data from other threads).
1074 * I don't lock, because I only depend on the fd being valid, and this won't be
1075 * true until a session setup is completed.
1077 static UCHAR NetBTSend(void *adapt, void *sess, PNCB ncb)
1079 NetBTAdapter *adapter = (NetBTAdapter *)adapt;
1080 NetBTSession *session = (NetBTSession *)sess;
1081 UCHAR buffer[NBSS_HDRSIZE], ret;
1086 TRACE("adapt %p, session %p, NCB %p\n", adapt, session, ncb);
1088 if (!adapter) return NRC_ENVNOTDEF;
1089 if (!ncb) return NRC_INVADDRESS;
1090 if (!ncb->ncb_buffer) return NRC_BADDR;
1091 if (!session) return NRC_SNUMOUT;
1092 if (session->fd == INVALID_SOCKET) return NRC_SNUMOUT;
1094 buffer[0] = NBSS_MSG;
1096 NBR_ADDWORD(&buffer[2], ncb->ncb_length);
1098 wsaBufs[0].len = NBSS_HDRSIZE;
1099 wsaBufs[0].buf = (char*)buffer;
1100 wsaBufs[1].len = ncb->ncb_length;
1101 wsaBufs[1].buf = (char*)ncb->ncb_buffer;
1103 r = WSASend(session->fd, wsaBufs, sizeof(wsaBufs) / sizeof(wsaBufs[0]),
1104 &bytesSent, 0, NULL, NULL);
1105 if (r == SOCKET_ERROR)
1107 NetBIOSHangupSession(ncb);
1110 else if (bytesSent < NBSS_HDRSIZE + ncb->ncb_length)
1112 FIXME("Only sent %d bytes (of %d), hanging up session\n", bytesSent,
1113 NBSS_HDRSIZE + ncb->ncb_length);
1114 NetBIOSHangupSession(ncb);
1120 adapter->xmit_success++;
1122 TRACE("returning 0x%02x\n", ret);
1126 static UCHAR NetBTRecv(void *adapt, void *sess, PNCB ncb)
1128 NetBTAdapter *adapter = (NetBTAdapter *)adapt;
1129 NetBTSession *session = (NetBTSession *)sess;
1130 UCHAR buffer[NBSS_HDRSIZE], ret;
1133 DWORD bufferCount, bytesReceived, flags;
1135 TRACE("adapt %p, session %p, NCB %p\n", adapt, session, ncb);
1137 if (!adapter) return NRC_ENVNOTDEF;
1138 if (!ncb) return NRC_BADDR;
1139 if (!ncb->ncb_buffer) return NRC_BADDR;
1140 if (!session) return NRC_SNUMOUT;
1141 if (session->fd == INVALID_SOCKET) return NRC_SNUMOUT;
1143 EnterCriticalSection(&session->cs);
1145 if (session->bytesPending == 0)
1148 wsaBufs[0].len = NBSS_HDRSIZE;
1149 wsaBufs[0].buf = (char*)buffer;
1151 wsaBufs[bufferCount].len = ncb->ncb_length;
1152 wsaBufs[bufferCount].buf = (char*)ncb->ncb_buffer;
1156 /* FIXME: should poll a bit so I can check the cancel flag */
1157 r = WSARecv(session->fd, wsaBufs, bufferCount, &bytesReceived, &flags,
1159 if (r == SOCKET_ERROR && WSAGetLastError() != WSAEWOULDBLOCK)
1161 LeaveCriticalSection(&session->cs);
1162 ERR("Receive error, WSAGetLastError() returns %d\n", WSAGetLastError());
1163 NetBIOSHangupSession(ncb);
1166 else if (NCB_CANCELLED(ncb))
1168 LeaveCriticalSection(&session->cs);
1173 if (bufferCount == 2)
1175 if (buffer[0] == NBSS_KEEPALIVE)
1177 LeaveCriticalSection(&session->cs);
1178 FIXME("Oops, received a session keepalive and lost my place\n");
1179 /* need to read another session header until we get a session
1180 * message header. */
1181 NetBIOSHangupSession(ncb);
1185 else if (buffer[0] != NBSS_MSG)
1187 LeaveCriticalSection(&session->cs);
1188 FIXME("Received unexpected session msg type %d\n", buffer[0]);
1189 NetBIOSHangupSession(ncb);
1195 if (buffer[1] & NBSS_EXTENSION)
1197 LeaveCriticalSection(&session->cs);
1198 FIXME("Received a message that's too long for my taste\n");
1199 NetBIOSHangupSession(ncb);
1205 session->bytesPending = NBSS_HDRSIZE
1206 + NBR_GETWORD(&buffer[2]) - bytesReceived;
1207 ncb->ncb_length = bytesReceived - NBSS_HDRSIZE;
1208 LeaveCriticalSection(&session->cs);
1214 if (bytesReceived < session->bytesPending)
1215 session->bytesPending -= bytesReceived;
1217 session->bytesPending = 0;
1218 LeaveCriticalSection(&session->cs);
1219 ncb->ncb_length = bytesReceived;
1221 if (session->bytesPending > 0)
1226 adapter->recv_success++;
1230 TRACE("returning 0x%02x\n", ret);
1234 static UCHAR NetBTHangup(void *adapt, void *sess)
1236 NetBTSession *session = (NetBTSession *)sess;
1238 TRACE("adapt %p, session %p\n", adapt, session);
1240 if (!session) return NRC_SNUMOUT;
1242 /* I don't lock the session, because NetBTRecv knows not to decrement
1243 * past 0, so if a receive completes after this it should still deal.
1245 closesocket(session->fd);
1246 session->fd = INVALID_SOCKET;
1247 session->bytesPending = 0;
1248 session->cs.DebugInfo->Spare[0] = 0;
1249 DeleteCriticalSection(&session->cs);
1250 HeapFree(GetProcessHeap(), 0, session);
1255 static void NetBTCleanupAdapter(void *adapt)
1257 TRACE("adapt %p\n", adapt);
1260 NetBTAdapter *adapter = (NetBTAdapter *)adapt;
1262 if (adapter->nameCache)
1263 NBNameCacheDestroy(adapter->nameCache);
1264 HeapFree(GetProcessHeap(), 0, adapt);
1268 static void NetBTCleanup(void)
1273 NBNameCacheDestroy(gNameCache);
1278 static UCHAR NetBTRegisterAdapter(const MIB_IPADDRROW *ipRow)
1281 NetBTAdapter *adapter;
1283 if (!ipRow) return NRC_BADDR;
1285 adapter = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(NetBTAdapter));
1288 adapter->ipr = *ipRow;
1289 if (!NetBIOSRegisterAdapter(gTransportID, ipRow->dwIndex, adapter))
1291 NetBTCleanupAdapter(adapter);
1298 ret = NRC_OSRESNOTAV;
1302 /* Callback for NetBIOS adapter enumeration. Assumes closure is a pointer to
1303 * a MIB_IPADDRTABLE containing all the IP adapters needed to be added to the
1304 * NetBIOS adapter table. For each callback, checks if the passed-in adapt
1305 * has an entry in the table; if so, this adapter was enumerated previously,
1306 * and it's enabled. As a flag, the table's dwAddr entry is changed to
1307 * INADDR_LOOPBACK, since this is an invalid address for a NetBT adapter.
1308 * The NetBTEnum function will add any remaining adapters from the
1309 * MIB_IPADDRTABLE to the NetBIOS adapter table.
1311 static BOOL NetBTEnumCallback(UCHAR totalLANAs, UCHAR lanaIndex,
1312 ULONG transport, const NetBIOSAdapterImpl *data, void *closure)
1315 PMIB_IPADDRTABLE table = (PMIB_IPADDRTABLE)closure;
1322 for (ndx = 0; !ret && ndx < table->dwNumEntries; ndx++)
1324 const NetBTAdapter *adapter = (const NetBTAdapter *)data->data;
1326 if (table->table[ndx].dwIndex == adapter->ipr.dwIndex)
1328 NetBIOSEnableAdapter(data->lana);
1329 table->table[ndx].dwAddr = INADDR_LOOPBACK;
1339 /* Enumerates adapters by:
1340 * - retrieving the IP address table for the local machine
1341 * - eliminating loopback addresses from the table
1342 * - eliminating redundant addresses, that is, multiple addresses on the same
1344 * Calls NetBIOSEnumAdapters, passing the resulting table as the callback
1345 * data. The callback reenables each adapter that's already in the NetBIOS
1346 * table. After NetBIOSEnumAdapters returns, this function adds any remaining
1347 * adapters to the NetBIOS table.
1349 static UCHAR NetBTEnum(void)
1356 if (GetIpAddrTable(NULL, &size, FALSE) == ERROR_INSUFFICIENT_BUFFER)
1358 PMIB_IPADDRTABLE ipAddrs, coalesceTable = NULL;
1359 DWORD numIPAddrs = (size - sizeof(MIB_IPADDRTABLE)) /
1360 sizeof(MIB_IPADDRROW) + 1;
1362 ipAddrs = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, size);
1364 coalesceTable = HeapAlloc(GetProcessHeap(),
1365 HEAP_ZERO_MEMORY, sizeof(MIB_IPADDRTABLE) +
1366 (min(numIPAddrs, MAX_LANA + 1) - 1) * sizeof(MIB_IPADDRROW));
1367 if (ipAddrs && coalesceTable)
1369 if (GetIpAddrTable(ipAddrs, &size, FALSE) == ERROR_SUCCESS)
1373 for (ndx = 0; ndx < ipAddrs->dwNumEntries; ndx++)
1375 if ((ipAddrs->table[ndx].dwAddr &
1376 ipAddrs->table[ndx].dwMask) !=
1377 htonl((INADDR_LOOPBACK & IN_CLASSA_NET)))
1379 BOOL newNetwork = TRUE;
1382 /* make sure we don't have more than one entry
1384 for (innerIndex = 0; newNetwork &&
1385 innerIndex < coalesceTable->dwNumEntries; innerIndex++)
1386 if ((ipAddrs->table[ndx].dwAddr &
1387 ipAddrs->table[ndx].dwMask) ==
1388 (coalesceTable->table[innerIndex].dwAddr
1389 & coalesceTable->table[innerIndex].dwMask))
1393 memcpy(&coalesceTable->table[
1394 coalesceTable->dwNumEntries++],
1395 &ipAddrs->table[ndx], sizeof(MIB_IPADDRROW));
1399 NetBIOSEnumAdapters(gTransportID, NetBTEnumCallback,
1402 for (ndx = 0; ret == NRC_GOODRET &&
1403 ndx < coalesceTable->dwNumEntries; ndx++)
1404 if (coalesceTable->table[ndx].dwAddr != INADDR_LOOPBACK)
1405 ret = NetBTRegisterAdapter(&coalesceTable->table[ndx]);
1409 HeapFree(GetProcessHeap(), 0, ipAddrs);
1410 HeapFree(GetProcessHeap(), 0, coalesceTable);
1413 ret = NRC_OSRESNOTAV;
1417 TRACE("returning 0x%02x\n", ret);
1421 static const WCHAR VxD_MSTCPW[] = { 'S','Y','S','T','E','M','\\','C','u','r',
1422 'r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\','S','e','r','v',
1423 'i','c','e','s','\\','V','x','D','\\','M','S','T','C','P','\0' };
1424 static const WCHAR NetBT_ParametersW[] = { 'S','Y','S','T','E','M','\\','C','u',
1425 'r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\','S','e','r',
1426 'v','i','c','e','s','\\','N','e','t','B','T','\\','P','a','r','a','m','e','t',
1428 static const WCHAR EnableDNSW[] = { 'E','n','a','b','l','e','D','N','S','\0' };
1429 static const WCHAR BcastNameQueryCountW[] = { 'B','c','a','s','t','N','a','m',
1430 'e','Q','u','e','r','y','C','o','u','n','t','\0' };
1431 static const WCHAR BcastNameQueryTimeoutW[] = { 'B','c','a','s','t','N','a','m',
1432 'e','Q','u','e','r','y','T','i','m','e','o','u','t','\0' };
1433 static const WCHAR NameSrvQueryCountW[] = { 'N','a','m','e','S','r','v',
1434 'Q','u','e','r','y','C','o','u','n','t','\0' };
1435 static const WCHAR NameSrvQueryTimeoutW[] = { 'N','a','m','e','S','r','v',
1436 'Q','u','e','r','y','T','i','m','e','o','u','t','\0' };
1437 static const WCHAR ScopeIDW[] = { 'S','c','o','p','e','I','D','\0' };
1438 static const WCHAR CacheTimeoutW[] = { 'C','a','c','h','e','T','i','m','e','o',
1440 static const WCHAR Config_NetworkW[] = { 'S','o','f','t','w','a','r','e','\\',
1441 'W','i','n','e','\\','N','e','t','w','o','r','k','\0' };
1443 /* Initializes global variables and registers the NetBT transport */
1444 void NetBTInit(void)
1447 NetBIOSTransport transport;
1453 gBCastQueries = BCAST_QUERIES;
1454 gBCastQueryTimeout = BCAST_QUERY_TIMEOUT;
1455 gWINSQueries = WINS_QUERIES;
1456 gWINSQueryTimeout = WINS_QUERY_TIMEOUT;
1457 gNumWINSServers = 0;
1458 memset(gWINSServers, 0, sizeof(gWINSServers));
1460 gCacheTimeout = CACHE_TIMEOUT;
1462 /* Try to open the Win9x NetBT configuration key */
1463 ret = RegOpenKeyExW(HKEY_LOCAL_MACHINE, VxD_MSTCPW, 0, KEY_READ, &hKey);
1464 /* If that fails, try the WinNT NetBT configuration key */
1465 if (ret != ERROR_SUCCESS)
1466 ret = RegOpenKeyExW(HKEY_LOCAL_MACHINE, NetBT_ParametersW, 0, KEY_READ,
1468 if (ret == ERROR_SUCCESS)
1472 size = sizeof(dword);
1473 if (RegQueryValueExW(hKey, EnableDNSW, NULL, NULL,
1474 (LPBYTE)&dword, &size) == ERROR_SUCCESS)
1476 size = sizeof(dword);
1477 if (RegQueryValueExW(hKey, BcastNameQueryCountW, NULL, NULL,
1478 (LPBYTE)&dword, &size) == ERROR_SUCCESS && dword >= MIN_QUERIES
1479 && dword <= MAX_QUERIES)
1480 gBCastQueries = dword;
1481 size = sizeof(dword);
1482 if (RegQueryValueExW(hKey, BcastNameQueryTimeoutW, NULL, NULL,
1483 (LPBYTE)&dword, &size) == ERROR_SUCCESS && dword >= MIN_QUERY_TIMEOUT
1484 && dword <= MAX_QUERY_TIMEOUT)
1485 gBCastQueryTimeout = dword;
1486 size = sizeof(dword);
1487 if (RegQueryValueExW(hKey, NameSrvQueryCountW, NULL, NULL,
1488 (LPBYTE)&dword, &size) == ERROR_SUCCESS && dword >= MIN_QUERIES
1489 && dword <= MAX_QUERIES)
1490 gWINSQueries = dword;
1491 size = sizeof(dword);
1492 if (RegQueryValueExW(hKey, NameSrvQueryTimeoutW, NULL, NULL,
1493 (LPBYTE)&dword, &size) == ERROR_SUCCESS && dword >= MIN_QUERY_TIMEOUT
1494 && dword <= MAX_QUERY_TIMEOUT)
1495 gWINSQueryTimeout = dword;
1496 size = MAX_DOMAIN_NAME_LEN - 1;
1497 if (RegQueryValueExW(hKey, ScopeIDW, NULL, NULL, (LPBYTE)gScopeID + 1, &size)
1500 /* convert into L2-encoded version, suitable for use by
1504 for (ptr = gScopeID + 1; *ptr &&
1505 ptr - gScopeID < MAX_DOMAIN_NAME_LEN; )
1507 for (lenPtr = ptr - 1, *lenPtr = 0; *ptr && *ptr != '.' &&
1508 ptr - gScopeID < MAX_DOMAIN_NAME_LEN; ptr++)
1513 if (RegQueryValueExW(hKey, CacheTimeoutW, NULL, NULL,
1514 (LPBYTE)&dword, &size) == ERROR_SUCCESS && dword >= MIN_CACHE_TIMEOUT)
1515 gCacheTimeout = dword;
1518 /* WINE-specific NetBT registry settings. Because our adapter naming is
1519 * different than MS', we can't do per-adapter WINS configuration in the
1520 * same place. Just do a global WINS configuration instead.
1522 /* @@ Wine registry key: HKCU\Software\Wine\Network */
1523 if (RegOpenKeyW(HKEY_CURRENT_USER, Config_NetworkW, &hKey) == ERROR_SUCCESS)
1525 static const char *nsValueNames[] = { "WinsServer", "BackupWinsServer" };
1529 for (ndx = 0; ndx < sizeof(nsValueNames) / sizeof(nsValueNames[0]);
1532 size = sizeof(nsString) / sizeof(char);
1533 if (RegQueryValueExA(hKey, nsValueNames[ndx], NULL, NULL,
1534 (LPBYTE)nsString, &size) == ERROR_SUCCESS)
1536 unsigned long addr = inet_addr(nsString);
1538 if (addr != INADDR_NONE && gNumWINSServers < MAX_WINS_SERVERS)
1539 gWINSServers[gNumWINSServers++] = addr;
1545 transport.enumerate = NetBTEnum;
1546 transport.astat = NetBTAstat;
1547 transport.findName = NetBTFindName;
1548 transport.call = NetBTCall;
1549 transport.send = NetBTSend;
1550 transport.recv = NetBTRecv;
1551 transport.hangup = NetBTHangup;
1552 transport.cleanupAdapter = NetBTCleanupAdapter;
1553 transport.cleanup = NetBTCleanup;
1554 memcpy(&gTransportID, TRANSPORT_NBT, sizeof(ULONG));
1555 NetBIOSRegisterTransport(gTransportID, &transport);