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.

2991 lines
87 KiB

  1. /*++
  2. Copyright (c) 1991-1992 Microsoft Corporation
  3. Module Name:
  4. srvenum.c
  5. Abstract:
  6. This module contains the worker routine for the NetServerEnum API
  7. implemented by the Workstation service.
  8. Author:
  9. Rita Wong (ritaw) 25-Mar-1991
  10. Revision History:
  11. --*/
  12. #include "precomp.h"
  13. #pragma hdrstop
  14. NET_API_STATUS NET_API_FUNCTION
  15. BrNetServerEnum(
  16. IN PNETWORK Network OPTIONAL,
  17. IN LPCWSTR ClientName OPTIONAL,
  18. IN ULONG Level,
  19. IN DWORD PreferedMaximumLength,
  20. OUT PVOID *Buffer,
  21. OUT LPDWORD EntriesRead,
  22. OUT LPDWORD TotalEntries,
  23. IN DWORD ServerType,
  24. IN LPCWSTR Domain,
  25. IN LPCWSTR FirstNameToReturn OPTIONAL
  26. );
  27. NET_API_STATUS
  28. BrRetrieveServerListForMaster(
  29. IN PNETWORK Network,
  30. IN OUT PVOID *Buffer,
  31. OUT PDWORD EntriesRead,
  32. OUT PDWORD TotalEntries,
  33. IN DWORD Level,
  34. IN DWORD ServerType,
  35. IN DWORD PreferedMaximumLength,
  36. IN BOOLEAN LocalListOnly,
  37. IN LPTSTR ClientName,
  38. IN LPTSTR DomainName,
  39. IN LPCWSTR FirstNameToReturn
  40. );
  41. NET_API_STATUS
  42. BrRetrieveServerListForBackup(
  43. IN PNETWORK Network,
  44. IN OUT PVOID *Buffer,
  45. OUT PDWORD EntriesRead,
  46. OUT PDWORD TotalEntries,
  47. IN DWORD Level,
  48. IN DWORD ServerType,
  49. IN DWORD PreferedMaximumLength,
  50. IN LPCWSTR FirstNameToReturn
  51. );
  52. //
  53. // This points to XsConvertServerEnumBuffer, which is dynamically loaded from
  54. // xactsrv.dll when required
  55. //
  56. XS_CONVERT_SERVER_ENUM_BUFFER_FUNCTION *pXsConvertServerEnumBuffer = NULL;
  57. NET_API_STATUS NET_API_FUNCTION
  58. I_BrowserrServerEnum(
  59. IN LPTSTR ServerName OPTIONAL,
  60. IN LPTSTR TransportName OPTIONAL,
  61. IN LPTSTR ClientName OPTIONAL,
  62. IN OUT LPSERVER_ENUM_STRUCT InfoStruct,
  63. IN DWORD PreferedMaximumLength,
  64. OUT LPDWORD TotalEntries,
  65. IN DWORD ServerType,
  66. IN LPTSTR Domain,
  67. IN OUT LPDWORD ResumeHandle OPTIONAL
  68. )
  69. /*++
  70. Routine Description:
  71. This function is the NetServerEnum entry point in the Workstation service.
  72. Arguments:
  73. ServerName - Supplies the name of server to execute this function
  74. TransportName - Supplies the name of xport on which to enumerate servers
  75. InfoStruct - This structure supplies the level of information requested,
  76. returns a pointer to the buffer allocated by the Workstation service
  77. which contains a sequence of information structure of the specified
  78. information level, and returns the number of entries read. The buffer
  79. pointer is set to NULL if return code is not NERR_Success or
  80. ERROR_MORE_DATA, or if EntriesRead returned is 0. The EntriesRead
  81. value is only valid if the return code is NERR_Success or
  82. ERROR_MORE_DATA.
  83. PreferedMaximumLength - Supplies the number of bytes of information
  84. to return in the buffer. If this value is MAXULONG, we will try
  85. to return all available information if there is enough memory
  86. resource.
  87. TotalEntries - Returns the total number of entries available. This value
  88. is returned only if the return code is NERR_Success or ERROR_MORE_DATA.
  89. ServerType - Supplies the type of server to enumerate.
  90. Domain - Supplies the name of one of the active domains to enumerate the
  91. servers from. If NULL, servers from the primary domain, logon domain
  92. and other domains are enumerated.
  93. ResumeHandle - Supplies and returns the point to continue with enumeration.
  94. Return Value:
  95. NET_API_STATUS - NERR_Success or reason for failure.
  96. --*/
  97. {
  98. NET_API_STATUS NetStatus;
  99. NetStatus = I_BrowserrServerEnumEx(
  100. ServerName,
  101. TransportName,
  102. ClientName,
  103. InfoStruct,
  104. PreferedMaximumLength,
  105. TotalEntries,
  106. ServerType,
  107. Domain,
  108. NULL ); // NULL FirstNameToReturn
  109. if (ARGUMENT_PRESENT(ResumeHandle)) {
  110. *ResumeHandle = 0;
  111. }
  112. return NetStatus;
  113. }
  114. NET_API_STATUS NET_API_FUNCTION
  115. I_BrowserrServerEnumEx(
  116. IN LPTSTR ServerName OPTIONAL,
  117. IN LPTSTR TransportName OPTIONAL,
  118. IN LPTSTR ClientName OPTIONAL,
  119. IN OUT LPSERVER_ENUM_STRUCT InfoStruct,
  120. IN DWORD PreferedMaximumLength,
  121. OUT LPDWORD TotalEntries,
  122. IN DWORD ServerType,
  123. IN LPTSTR Domain,
  124. IN LPTSTR FirstNameToReturnArg
  125. )
  126. /*++
  127. Routine Description:
  128. This function is the NetServerEnum entry point in the Workstation service.
  129. Arguments:
  130. ServerName - Supplies the name of server to execute this function
  131. TransportName - Supplies the name of xport on which to enumerate servers
  132. InfoStruct - This structure supplies the level of information requested,
  133. returns a pointer to the buffer allocated by the Workstation service
  134. which contains a sequence of information structure of the specified
  135. information level, and returns the number of entries read. The buffer
  136. pointer is set to NULL if return code is not NERR_Success or
  137. ERROR_MORE_DATA, or if EntriesRead returned is 0. The EntriesRead
  138. value is only valid if the return code is NERR_Success or
  139. ERROR_MORE_DATA.
  140. PreferedMaximumLength - Supplies the number of bytes of information
  141. to return in the buffer. If this value is MAXULONG, we will try
  142. to return all available information if there is enough memory
  143. resource.
  144. TotalEntries - Returns the total number of entries available. This value
  145. is returned only if the return code is NERR_Success or ERROR_MORE_DATA.
  146. ServerType - Supplies the type of server to enumerate.
  147. Domain - Supplies the name of one of the active domains to enumerate the
  148. servers from. If NULL, servers from the primary domain, logon domain
  149. and other domains are enumerated.
  150. FirstNameToReturnArg - Supplies the name of the first domain or server entry to return.
  151. The caller can use this parameter to implement a resume handle of sorts by passing
  152. the name of the last entry returned on a previous call. (Notice that the specified
  153. entry will, also, be returned on this call unless it has since been deleted.)
  154. Pass NULL to start with the first entry available.
  155. Return Value:
  156. NET_API_STATUS - NERR_Success or reason for failure.
  157. --*/
  158. {
  159. NET_API_STATUS status;
  160. PVOID Buffer = NULL;
  161. ULONG EntriesRead;
  162. BOOLEAN NetworkLocked = FALSE;
  163. PNETWORK Network = NULL;
  164. UNICODE_STRING NetworkName;
  165. WCHAR FirstNameToReturn[DNLEN+1];
  166. PDOMAIN_INFO DomainInfo = NULL;
  167. #if DBG
  168. DWORD StartTickCount, EndTickCount;
  169. #endif
  170. UNREFERENCED_PARAMETER(ServerName);
  171. #if DBG
  172. StartTickCount = GetTickCount();
  173. #endif
  174. if (!ARGUMENT_PRESENT(TransportName)) {
  175. status = ERROR_INVALID_PARAMETER;
  176. goto Cleanup;
  177. }
  178. if (!ARGUMENT_PRESENT(InfoStruct)) {
  179. status = ERROR_INVALID_PARAMETER;
  180. goto Cleanup;
  181. }
  182. if ( InfoStruct->Level == 101 &&
  183. !InfoStruct->ServerInfo.Level101 ) {
  184. status = ERROR_INVALID_PARAMETER;
  185. goto Cleanup;
  186. } else if ( InfoStruct->Level == 100 &&
  187. !InfoStruct->ServerInfo.Level100 ) {
  188. status = ERROR_INVALID_PARAMETER;
  189. goto Cleanup;
  190. }
  191. #ifdef ENABLE_PSEUDO_BROWSER
  192. //
  193. // Pseudo Server returns nothing
  194. //
  195. if (BrInfo.PseudoServerLevel == BROWSER_PSEUDO) {
  196. if (InfoStruct->Level == 101) {
  197. InfoStruct->ServerInfo.Level101->Buffer = NULL;
  198. InfoStruct->ServerInfo.Level101->EntriesRead = 0;
  199. } else {
  200. InfoStruct->ServerInfo.Level100->Buffer = NULL;
  201. InfoStruct->ServerInfo.Level100->EntriesRead = 0;
  202. }
  203. status = ERROR_SUCCESS;
  204. goto Cleanup;
  205. }
  206. #endif
  207. //
  208. // Find the requested domain.
  209. //
  210. DomainInfo = BrFindDomain( Domain, TRUE );
  211. if ( DomainInfo == NULL) {
  212. status = ERROR_NO_SUCH_DOMAIN;
  213. goto Cleanup;
  214. }
  215. //
  216. // Find the requested network
  217. //
  218. RtlInitUnicodeString(&NetworkName, TransportName);
  219. Network = BrFindNetwork( DomainInfo, &NetworkName);
  220. if (Network == NULL) {
  221. BrPrint(( BR_CRITICAL,
  222. "%ws: %ws: BrowserrServerEnum: Network not found.\n",
  223. Domain,
  224. TransportName));
  225. status = ERROR_FILE_NOT_FOUND;
  226. goto Cleanup;
  227. }
  228. //
  229. // If the caller has asked us to use the alternate transport,
  230. // do so.
  231. //
  232. if ((ServerType != SV_TYPE_ALL) &&
  233. (ServerType & SV_TYPE_ALTERNATE_XPORT) ) {
  234. //
  235. // If this transport has an alternate network, then actually
  236. // query the alternate name, not the real one.
  237. //
  238. if (Network->AlternateNetwork != NULL) {
  239. PNETWORK TempNetwork = Network;
  240. Network = Network->AlternateNetwork;
  241. if ( !BrReferenceNetwork( Network )) {
  242. Network = TempNetwork;
  243. status = ERROR_INVALID_PARAMETER;
  244. goto Cleanup;
  245. }
  246. BrDereferenceNetwork( TempNetwork );
  247. } else {
  248. status = ERROR_INVALID_PARAMETER;
  249. goto Cleanup;
  250. }
  251. ServerType &= ~SV_TYPE_ALTERNATE_XPORT;
  252. if (ServerType == 0) {
  253. ServerType = SV_TYPE_ALL;
  254. }
  255. }
  256. if (!LOCK_NETWORK_SHARED(Network)) {
  257. status = NERR_InternalError;
  258. goto Cleanup;
  259. }
  260. NetworkLocked = TRUE;
  261. if (!(Network->Role & (ROLE_BACKUP | ROLE_MASTER))) {
  262. //
  263. // If this is a wannish transport,
  264. // and the caller is asking for "local list",
  265. // try to find another wannish transport that is a master browser.
  266. //
  267. // The domain master browser doesn't have any control over which
  268. // transport he comes in on. If he picks a disabled transport,
  269. // this code will find the enabled transport.
  270. //
  271. // There are cases where there is more than one wannish master browser
  272. // transport on this machine. In that case, BrNetServerEnum merges the
  273. // local lists for all wannish transports to ensure that all list are returned
  274. // to the domain master browser
  275. //
  276. if ( (Network->Flags & NETWORK_WANNISH) != 0 &&
  277. (ServerType == SV_TYPE_LOCAL_LIST_ONLY ||
  278. ServerType == (SV_TYPE_LOCAL_LIST_ONLY|SV_TYPE_DOMAIN_ENUM) ) ) {
  279. PNETWORK TempNetwork;
  280. TempNetwork = BrFindWannishMasterBrowserNetwork( DomainInfo );
  281. if ( TempNetwork == NULL ) {
  282. BrPrint(( BR_CRITICAL,
  283. "%ws: %ws: Browse request received from %ws, but not backup or master\n",
  284. Network->DomainInfo->DomUnicodeDomainName,
  285. Network->NetworkName.Buffer,
  286. ClientName));
  287. status = ERROR_REQ_NOT_ACCEP;
  288. goto Cleanup;
  289. }
  290. //
  291. // Ditch the old network.
  292. //
  293. UNLOCK_NETWORK(Network);
  294. NetworkLocked = FALSE;
  295. BrDereferenceNetwork( Network );
  296. Network = TempNetwork;
  297. //
  298. // Use the new network
  299. //
  300. if (!LOCK_NETWORK_SHARED(Network)) {
  301. status = NERR_InternalError;
  302. goto Cleanup;
  303. }
  304. NetworkLocked = TRUE;
  305. BrPrint(( BR_SERVER_ENUM,
  306. "%ws: %ws: Is wannish IP Network found for %ws\n",
  307. Network->DomainInfo->DomUnicodeDomainName,
  308. Network->NetworkName.Buffer,
  309. ClientName));
  310. } else {
  311. BrPrint(( BR_CRITICAL,
  312. "%ws: %ws: Browse request received from %ws, but not backup or master\n",
  313. Network->DomainInfo->DomUnicodeDomainName,
  314. Network->NetworkName.Buffer,
  315. ClientName));
  316. status = ERROR_REQ_NOT_ACCEP;
  317. goto Cleanup;
  318. }
  319. }
  320. //
  321. // Canonicalize the FirstNameToReturn.
  322. //
  323. if (ARGUMENT_PRESENT(FirstNameToReturnArg) && *FirstNameToReturnArg != L'\0') {
  324. if ( I_NetNameCanonicalize(
  325. NULL,
  326. (LPWSTR) FirstNameToReturnArg,
  327. FirstNameToReturn,
  328. sizeof(FirstNameToReturn),
  329. NAMETYPE_WORKGROUP,
  330. LM2X_COMPATIBLE
  331. ) != NERR_Success) {
  332. status = ERROR_INVALID_PARAMETER;
  333. goto Cleanup;
  334. }
  335. } else {
  336. FirstNameToReturn[0] = L'\0';
  337. }
  338. status = BrNetServerEnum(Network,
  339. ClientName,
  340. InfoStruct->Level,
  341. PreferedMaximumLength,
  342. &Buffer,
  343. &EntriesRead,
  344. TotalEntries,
  345. ServerType,
  346. Domain,
  347. FirstNameToReturn );
  348. //
  349. // Return output parameters other than output buffer from request packet.
  350. //
  351. if (status == NERR_Success || status == ERROR_MORE_DATA) {
  352. if (InfoStruct->Level == 101) {
  353. InfoStruct->ServerInfo.Level101->Buffer = (PSERVER_INFO_101) Buffer;
  354. InfoStruct->ServerInfo.Level101->EntriesRead = EntriesRead;
  355. } else {
  356. InfoStruct->ServerInfo.Level100->Buffer = (PSERVER_INFO_100) Buffer;
  357. InfoStruct->ServerInfo.Level100->EntriesRead = EntriesRead;
  358. }
  359. }
  360. Cleanup:
  361. if (NetworkLocked) {
  362. UNLOCK_NETWORK(Network);
  363. }
  364. if ( Network != NULL ) {
  365. #if DBG
  366. EndTickCount = GetTickCount();
  367. BrPrint(( BR_SERVER_ENUM,
  368. "%ws: %ws: Browse request for %ws took %ld milliseconds\n",
  369. Network->DomainInfo->DomUnicodeDomainName,
  370. Network->NetworkName.Buffer,
  371. ClientName,
  372. EndTickCount - StartTickCount));
  373. #endif
  374. BrDereferenceNetwork( Network );
  375. }
  376. if ( DomainInfo != NULL ) {
  377. BrDereferenceDomain( DomainInfo );
  378. }
  379. return status;
  380. }
  381. WORD
  382. I_BrowserServerEnumForXactsrv(
  383. IN LPCWSTR TransportName OPTIONAL,
  384. IN LPCWSTR ClientName OPTIONAL,
  385. IN ULONG Level,
  386. IN USHORT ClientLevel,
  387. IN PVOID ClientBuffer,
  388. IN WORD BufferLength,
  389. IN DWORD PreferedMaximumLength,
  390. OUT LPDWORD EntriesFilled,
  391. OUT LPDWORD TotalEntries,
  392. IN DWORD ServerType,
  393. IN LPCWSTR Domain,
  394. IN LPCWSTR FirstNameToReturnArg OPTIONAL,
  395. OUT PWORD Converter
  396. )
  397. /*++
  398. Routine Description:
  399. This function is a private entrypoint for Xactsrv that bypasses RPC
  400. entirely.
  401. Arguments:
  402. TransportName - Supplies the name of xport on which to enumerate servers
  403. ClientName - Supplies the name of the client that requested the data
  404. Level - Level of data requested.
  405. ClientLevel - Level requested by the client.
  406. ClientBuffer - Output buffer allocated to hold the buffer.
  407. BufferLength - Size of ClientBuffer
  408. PreferedMaximumLength - Prefered maximum size of Client buffer if we are
  409. going to use the NT form of the buffer
  410. OUT LPDWORD EntriesFilled - The entries packed into ClientBuffer
  411. OUT LPDWORD TotalEntries - The total # of entries available.
  412. IN DWORD ServerType - Server type mask.
  413. IN LPTSTR Domain - Domain to query
  414. OUT PWORD Converter - Magic constant from Xactsrv that allows the client
  415. to convert the response buffer.
  416. Return Value:
  417. WORD - NERR_Success or reason for failure.
  418. --*/
  419. {
  420. NET_API_STATUS status;
  421. BOOLEAN networkLocked = FALSE;
  422. PNETWORK network = NULL;
  423. UNICODE_STRING networkName;
  424. PDOMAIN_INFO DomainInfo = NULL;
  425. PVOID buffer = NULL;
  426. DWORD entriesRead;
  427. PCACHED_BROWSE_RESPONSE response = NULL;
  428. WCHAR FirstNameToReturn[DNLEN+1];
  429. #if DBG
  430. DWORD startTickCount, endTickCount;
  431. startTickCount = GetTickCount();
  432. #endif
  433. try {
  434. //
  435. // If the browser isn't up and we get one of these calls, fail the API
  436. // immediately.
  437. //
  438. if (BrGlobalData.Status.dwCurrentState != SERVICE_RUNNING) {
  439. BrPrint(( BR_CRITICAL,
  440. "Browse request from %ws received, but browser not running\n", ClientName));
  441. try_return(status = NERR_ServiceNotInstalled);
  442. }
  443. if (!ARGUMENT_PRESENT(TransportName)) {
  444. try_return(status = ERROR_INVALID_PARAMETER);
  445. }
  446. //
  447. // Find the requested domain.
  448. //
  449. DomainInfo = BrFindDomain( (LPWSTR) Domain, TRUE );
  450. if ( DomainInfo == NULL) {
  451. try_return(status = ERROR_NO_SUCH_DOMAIN);
  452. }
  453. //
  454. // Look up the transport.
  455. //
  456. RtlInitUnicodeString(&networkName, TransportName);
  457. BrPrint(( BR_SERVER_ENUM,
  458. "%ws: %ws: NetServerEnum: Look up network for %ws\n",
  459. Domain,
  460. TransportName,
  461. ClientName));
  462. network = BrFindNetwork( DomainInfo, &networkName);
  463. //
  464. // If it returns NULL, the network was not found.
  465. // In that case just pick a network randomly and enumerate for that network
  466. // This is a workaround for RAID bug 614688.
  467. // The situation is that the request comes over NetbiosSMB and so
  468. // networkName points to that, and browser is not registered on that transport.
  469. //
  470. if ( network == NULL) {
  471. EnterCriticalSection(&NetworkCritSect);
  472. if ( ServicedNetworks.Flink != &ServicedNetworks ) {
  473. // If the list is not empty, just pick the first network
  474. network = CONTAINING_RECORD(ServicedNetworks.Flink, NETWORK, NextNet);
  475. network->ReferenceCount ++;
  476. }
  477. LeaveCriticalSection(&NetworkCritSect);
  478. }
  479. if (network == NULL) {
  480. BrPrint(( BR_CRITICAL,
  481. "%ws: %ws: Network not found.\n",
  482. Domain,
  483. TransportName));
  484. try_return(status = ERROR_FILE_NOT_FOUND);
  485. }
  486. //
  487. // If the caller has asked us to use the alternate transport,
  488. // do so.
  489. //
  490. if ((ServerType != SV_TYPE_ALL) &&
  491. (ServerType & SV_TYPE_ALTERNATE_XPORT) ) {
  492. //
  493. // If this transport has an alternate network, then actually
  494. // query the alternate name, not the real one.
  495. //
  496. if (network->AlternateNetwork != NULL) {
  497. PNETWORK TempNetwork = network;
  498. network = network->AlternateNetwork;
  499. if ( !BrReferenceNetwork( network )) {
  500. network = TempNetwork;
  501. try_return(status = ERROR_INVALID_PARAMETER);
  502. }
  503. BrDereferenceNetwork( TempNetwork );
  504. } else {
  505. try_return(status = ERROR_INVALID_PARAMETER);
  506. }
  507. ServerType &= ~SV_TYPE_ALTERNATE_XPORT;
  508. if (ServerType == 0) {
  509. ServerType = SV_TYPE_ALL;
  510. }
  511. }
  512. if (!LOCK_NETWORK_SHARED(network)) {
  513. try_return(status = NERR_InternalError);
  514. }
  515. networkLocked = TRUE;
  516. BrPrint(( BR_SERVER_ENUM,
  517. "%ws: %ws: Network found for %ws\n",
  518. network->DomainInfo->DomUnicodeDomainName,
  519. network->NetworkName.Buffer,
  520. ClientName));
  521. if (!(network->Role & (ROLE_BACKUP | ROLE_MASTER))) {
  522. //
  523. // If this is a wannish transport,
  524. // and the caller is asking for "local list",
  525. // try to find another wannish transport that is a master browser.
  526. //
  527. // The domain master browser doesn't have any control over which
  528. // transport he comes in on. If he picks a disabled transport,
  529. // this code will find the enabled transport.
  530. //
  531. // There are cases where there is more than one wannish master browser
  532. // transport on this machine. In that case, BrNetServerEnum merges the
  533. // local lists for all wannish transports to ensure that all list are returned
  534. // to the domain master browser
  535. //
  536. if ( (network->Flags & NETWORK_WANNISH) != 0 &&
  537. (ServerType == SV_TYPE_LOCAL_LIST_ONLY ||
  538. ServerType == (SV_TYPE_LOCAL_LIST_ONLY|SV_TYPE_DOMAIN_ENUM) ) ) {
  539. PNETWORK TempNetwork;
  540. TempNetwork = BrFindWannishMasterBrowserNetwork( DomainInfo );
  541. if ( TempNetwork == NULL ) {
  542. BrPrint(( BR_CRITICAL,
  543. "%ws: %ws: Browse request received from %ws, but not backup or master\n",
  544. network->DomainInfo->DomUnicodeDomainName,
  545. network->NetworkName.Buffer,
  546. ClientName));
  547. try_return(status = ERROR_REQ_NOT_ACCEP);
  548. }
  549. //
  550. // Ditch the old network.
  551. //
  552. UNLOCK_NETWORK(network);
  553. networkLocked = FALSE;
  554. BrDereferenceNetwork( network );
  555. network = TempNetwork;
  556. //
  557. // Use the new network
  558. //
  559. if (!LOCK_NETWORK_SHARED(network)) {
  560. try_return(status = NERR_InternalError);
  561. }
  562. networkLocked = TRUE;
  563. BrPrint(( BR_SERVER_ENUM,
  564. "%ws: %ws: Is wannish IP Network found for %ws\n",
  565. network->DomainInfo->DomUnicodeDomainName,
  566. network->NetworkName.Buffer,
  567. ClientName));
  568. } else {
  569. BrPrint(( BR_CRITICAL,
  570. "%ws: %ws: Browse request received from %ws, but not backup or master\n",
  571. network->DomainInfo->DomUnicodeDomainName,
  572. network->NetworkName.Buffer,
  573. ClientName));
  574. try_return(status = ERROR_REQ_NOT_ACCEP);
  575. }
  576. }
  577. //
  578. // If we weren't able to find a cached response buffer, or
  579. // if the cached response didn't have a buffer allocated for it,
  580. // we need to process the browse request.
  581. //
  582. if (network->Role & ROLE_MASTER) {
  583. //
  584. // Check to see if we should flush the cache at this time. If
  585. // we should, we need to release the network and re-acquire it
  586. // exclusively, because we use the network lock to protect
  587. // enumerations from flushes.
  588. //
  589. if ((BrCurrentSystemTime() - network->TimeCacheFlushed) > BrInfo.DriverQueryFrequency) {
  590. UNLOCK_NETWORK(network);
  591. networkLocked = FALSE;
  592. if (!LOCK_NETWORK(network)) {
  593. try_return(status = NERR_InternalError);
  594. }
  595. networkLocked = TRUE;
  596. //
  597. // We're running on a master browser, and we found a cached browse
  598. // request. See if we can safely use this cached value, or if we
  599. // should go to the driver for it.
  600. //
  601. EnterCriticalSection(&network->ResponseCacheLock);
  602. if ((BrCurrentSystemTime() - network->TimeCacheFlushed) > BrInfo.DriverQueryFrequency) {
  603. BrPrint(( BR_SERVER_ENUM,
  604. "%ws: %ws: Flushing cache\n",
  605. network->DomainInfo->DomUnicodeDomainName,
  606. network->NetworkName.Buffer ));
  607. network->TimeCacheFlushed = BrCurrentSystemTime();
  608. BrAgeResponseCache( network );
  609. }
  610. LeaveCriticalSection(&network->ResponseCacheLock);
  611. }
  612. }
  613. //
  614. // Canonicalize the FirstNameToReturn.
  615. //
  616. if (ARGUMENT_PRESENT(FirstNameToReturnArg) && *FirstNameToReturnArg != L'\0') {
  617. if ( I_NetNameCanonicalize(
  618. NULL,
  619. (LPWSTR) FirstNameToReturnArg,
  620. FirstNameToReturn,
  621. sizeof(FirstNameToReturn),
  622. NAMETYPE_WORKGROUP,
  623. LM2X_COMPATIBLE
  624. ) != NERR_Success) {
  625. try_return(status = ERROR_INVALID_PARAMETER);
  626. }
  627. } else {
  628. FirstNameToReturn[0] = L'\0';
  629. }
  630. if (!ARGUMENT_PRESENT(Domain) ||
  631. !STRICMP(Domain, network->DomainInfo->DomUnicodeDomainName)) {
  632. BrPrint(( BR_SERVER_ENUM,
  633. "%ws: %ws: Look up 0x%x/%d/%x.\n",
  634. Domain, TransportName, ServerType, ClientLevel, BufferLength));
  635. //
  636. // This request is for our primary domain. Look up a cached response
  637. // entry. If none is found for this request, allocate a new one and
  638. // return it.
  639. //
  640. response = BrLookupAndAllocateCachedEntry(
  641. network,
  642. ServerType,
  643. BufferLength,
  644. ClientLevel,
  645. FirstNameToReturn
  646. );
  647. }
  648. EnterCriticalSection(&network->ResponseCacheLock);
  649. if ((response == NULL)
  650. ||
  651. (response->Buffer == NULL)) {
  652. LeaveCriticalSection(&network->ResponseCacheLock);
  653. BrPrint(( BR_SERVER_ENUM,
  654. "%ws: %ws: Cached entry not found, or hit count too low. Retrieve actual list for %ws\n",
  655. Domain, TransportName, ClientName));
  656. status = BrNetServerEnum(network,
  657. ClientName,
  658. Level,
  659. PreferedMaximumLength,
  660. &buffer,
  661. &entriesRead,
  662. TotalEntries,
  663. ServerType,
  664. Domain,
  665. FirstNameToReturn
  666. );
  667. if (status == NERR_Success || status == ERROR_MORE_DATA) {
  668. BrPrint(( BR_SERVER_ENUM,
  669. "%ws: %ws: Convert NT buffer to Xactsrv buffer for %ws\n",
  670. Domain,
  671. TransportName,
  672. ClientName ));
  673. if( pXsConvertServerEnumBuffer == NULL ) {
  674. //
  675. // It doesn't really matter if several threads do this simultaneously.
  676. // We are never going to unload this library anyway. But we are delaying
  677. // the load of the library until now to speed up system boot and init.
  678. //
  679. HMODULE hLibrary = LoadLibrary( L"xactsrv.dll" );
  680. if( hLibrary != NULL ) {
  681. pXsConvertServerEnumBuffer = (XS_CONVERT_SERVER_ENUM_BUFFER_FUNCTION *)GetProcAddress(
  682. hLibrary, "XsConvertServerEnumBuffer" );
  683. }
  684. }
  685. if( pXsConvertServerEnumBuffer != NULL ) {
  686. status = pXsConvertServerEnumBuffer(
  687. buffer,
  688. entriesRead,
  689. TotalEntries,
  690. ClientLevel,
  691. ClientBuffer,
  692. BufferLength,
  693. EntriesFilled,
  694. Converter);
  695. } else {
  696. status = GetLastError();
  697. }
  698. if (status == NERR_Success || status == ERROR_MORE_DATA) {
  699. BrPrint(( BR_SERVER_ENUM,
  700. "%ws: %ws: Conversion done for %ws\n",
  701. Domain,
  702. TransportName,
  703. ClientName ));
  704. EnterCriticalSection(&network->ResponseCacheLock);
  705. if ((response != NULL) &&
  706. (response->Buffer == NULL) &&
  707. (response->TotalHitCount >= BrInfo.CacheHitLimit)) {
  708. BrPrint(( BR_SERVER_ENUM,
  709. "%ws: %ws: Save contents of server list for 0x%x/%d/%x.\n",
  710. Domain,
  711. TransportName,
  712. ServerType, ClientLevel, BufferLength));
  713. response->Buffer = MIDL_user_allocate(BufferLength);
  714. if ( response->Buffer != NULL ) {
  715. //
  716. // We successfully allocated the buffer, now copy
  717. // our response into the buffer and save away the
  718. // other useful stuff about the request.
  719. //
  720. RtlCopyMemory(response->Buffer, ClientBuffer, BufferLength);
  721. response->EntriesRead = *EntriesFilled;
  722. response->TotalEntries = *TotalEntries;
  723. response->Converter = *Converter;
  724. response->Status = (WORD)status;
  725. }
  726. }
  727. LeaveCriticalSection(&network->ResponseCacheLock);
  728. }
  729. }
  730. } else {
  731. ASSERT (response);
  732. ASSERT (response->Buffer);
  733. ASSERT (response->Size == BufferLength);
  734. BrPrint(( BR_SERVER_ENUM,
  735. "Cache hit. Use contents of server list for 0x%x/%d/%x.\n",
  736. Domain,
  737. TransportName,
  738. ServerType, ClientLevel, BufferLength));
  739. //
  740. // Copy over the cached response from the response cache into the
  741. // users response buffer.
  742. //
  743. RtlCopyMemory(ClientBuffer, response->Buffer, BufferLength);
  744. //
  745. // Also copy over the other useful stuff that the client will
  746. // want to know about.
  747. //
  748. *EntriesFilled = response->EntriesRead;
  749. *TotalEntries = response->TotalEntries;
  750. *Converter = response->Converter;
  751. status = response->Status;
  752. LeaveCriticalSection(&network->ResponseCacheLock);
  753. }
  754. try_exit:NOTHING;
  755. } finally {
  756. if (networkLocked) {
  757. UNLOCK_NETWORK(network);
  758. }
  759. //
  760. // If a buffer was allocated, free it.
  761. //
  762. if (buffer != NULL) {
  763. MIDL_user_free(buffer);
  764. }
  765. if ( network != NULL ) {
  766. #if DBG
  767. endTickCount = GetTickCount();
  768. BrPrint(( BR_SERVER_ENUM,
  769. "%ws: %ws: Browse request for %ws took %ld milliseconds\n",
  770. network->DomainInfo->DomUnicodeDomainName,
  771. network->NetworkName.Buffer,
  772. ClientName,
  773. endTickCount - startTickCount));
  774. #endif
  775. BrDereferenceNetwork( network );
  776. }
  777. if ( DomainInfo != NULL ) {
  778. BrDereferenceDomain( DomainInfo );
  779. }
  780. }
  781. return (WORD)status;
  782. }
  783. NET_API_STATUS NET_API_FUNCTION
  784. BrNetServerEnum(
  785. IN PNETWORK Network OPTIONAL,
  786. IN LPCWSTR ClientName OPTIONAL,
  787. IN ULONG Level,
  788. IN DWORD PreferedMaximumLength,
  789. OUT PVOID *Buffer,
  790. OUT LPDWORD EntriesRead,
  791. OUT LPDWORD TotalEntries,
  792. IN DWORD ServerType,
  793. IN LPCWSTR Domain,
  794. IN LPCWSTR FirstNameToReturn OPTIONAL
  795. )
  796. /*++
  797. Routine Description:
  798. This function is the real worker for the NetServerEnum entry point in the
  799. Workstation service.
  800. Arguments:
  801. ServerName - Supplies the name of server to execute this function
  802. TransportName - Supplies the name of xport on which to enumerate servers
  803. InfoStruct - This structure supplies the level of information requested,
  804. returns a pointer to the buffer allocated by the Workstation service
  805. which contains a sequence of information structure of the specified
  806. information level, and returns the number of entries read. The buffer
  807. pointer is set to NULL if return code is not NERR_Success or
  808. ERROR_MORE_DATA, or if EntriesRead returned is 0. The EntriesRead
  809. value is only valid if the return code is NERR_Success or
  810. ERROR_MORE_DATA.
  811. PreferedMaximumLength - Supplies the number of bytes of information
  812. to return in the buffer. If this value is MAXULONG, we will try
  813. to return all available information if there is enough memory
  814. resource.
  815. TotalEntries - Returns the total number of entries available. This value
  816. is returned only if the return code is NERR_Success or ERROR_MORE_DATA.
  817. ServerType - Supplies the type of server to enumerate.
  818. Domain - Supplies the name of one of the active domains to enumerate the
  819. servers from. If NULL, servers from the primary domain, logon domain
  820. and other domains are enumerated.
  821. FirstNameToReturn - Supplies the name of the first server or domain to return
  822. to the caller. If NULL, enumeration begins with the very first entry.
  823. Passed name must be the canonical form of the name.
  824. Return Value:
  825. NET_API_STATUS - NERR_Success or reason for failure.
  826. --*/
  827. {
  828. NET_API_STATUS status;
  829. DWORD DomainNameSize = 0;
  830. TCHAR DomainName[DNLEN + 1];
  831. BOOLEAN NetworkLocked = TRUE;
  832. BOOLEAN LocalListOnly = FALSE;
  833. PVOID DoubleHopLocalList = NULL;
  834. BrPrint(( BR_SERVER_ENUM,
  835. "%ws: %ws: Retrieve browse list for %lx for client %ws\n",
  836. Domain,
  837. Network->NetworkName.Buffer,
  838. ServerType,
  839. ClientName));
  840. EnterCriticalSection(&BrowserStatisticsLock);
  841. if (ServerType == SV_TYPE_DOMAIN_ENUM) {
  842. NumberOfDomainEnumerations += 1;
  843. } else if (ServerType == SV_TYPE_ALL) {
  844. NumberOfServerEnumerations += 1;
  845. } else {
  846. NumberOfOtherEnumerations += 1;
  847. }
  848. LeaveCriticalSection(&BrowserStatisticsLock);
  849. //
  850. // Only levels 100 and 101 are valid
  851. //
  852. if ((Level != 100) && (Level != 101)) {
  853. return ERROR_INVALID_LEVEL;
  854. }
  855. #ifdef ENABLE_PSEUDO_BROWSER
  856. //
  857. // Pseudo Server shortcut
  858. //
  859. if ( BrInfo.PseudoServerLevel == BROWSER_PSEUDO ) {
  860. *Buffer = NULL;
  861. *EntriesRead = 0;
  862. *TotalEntries = 0;
  863. return NERR_Success;
  864. }
  865. #endif
  866. if (ARGUMENT_PRESENT(Domain)) {
  867. //
  868. // NAMETYPE_WORKGROUP allows spaces.
  869. //
  870. if ( I_NetNameCanonicalize(
  871. NULL,
  872. (LPWSTR) Domain,
  873. DomainName,
  874. (DNLEN + 1) * sizeof(TCHAR),
  875. NAMETYPE_WORKGROUP,
  876. 0
  877. ) != NERR_Success) {
  878. return ERROR_INVALID_DOMAINNAME;
  879. }
  880. DomainNameSize = STRLEN(DomainName) * sizeof(WCHAR);
  881. }
  882. try {
  883. if (ServerType != SV_TYPE_ALL) {
  884. //
  885. // If the user has specified SV_TYPE_LOCAL_LIST_ONLY, we
  886. // will only accept SV_TYPE_DOMAIN_ENUM or 0 as legal bits
  887. // otherwise, we return an error.
  888. //
  889. if (ServerType & SV_TYPE_LOCAL_LIST_ONLY) {
  890. LocalListOnly = TRUE;
  891. // BrPrint(( BR_SERVER_ENUM, "Retrieve local list for %ws\n", ClientName));
  892. //
  893. // Turn off the LOCAL_LIST_ONLY bit.
  894. //
  895. ServerType &= ~SV_TYPE_LOCAL_LIST_ONLY;
  896. //
  897. // Check the remaining bits. The only two legal values
  898. // are SV_TYPE_DOMAIN_ENUM and 0
  899. //
  900. if (ServerType != SV_TYPE_DOMAIN_ENUM) {
  901. if (ServerType == 0) {
  902. ServerType = SV_TYPE_ALL;
  903. } else {
  904. try_return(status = ERROR_INVALID_FUNCTION);
  905. }
  906. }
  907. //
  908. // If we aren't a master browser, blow this request
  909. // off, since we don't have a local list.
  910. //
  911. if (!(Network->Role & ROLE_MASTER)) {
  912. BrPrint(( BR_CRITICAL,
  913. "%ws: %ws: Local list request received from %ws, but not master\n",
  914. Domain,
  915. Network->NetworkName.Buffer,
  916. ClientName));
  917. try_return(status = ERROR_REQ_NOT_ACCEP);
  918. }
  919. } else if (ServerType & SV_TYPE_DOMAIN_ENUM) {
  920. if (ServerType != SV_TYPE_DOMAIN_ENUM) {
  921. try_return(status = ERROR_INVALID_FUNCTION);
  922. }
  923. }
  924. }
  925. if (ARGUMENT_PRESENT(Domain) &&
  926. STRICMP(Domain, Network->DomainInfo->DomUnicodeDomainName)) {
  927. PINTERIM_ELEMENT DomainEntry;
  928. LPWSTR MasterName = NULL;
  929. BrPrint(( BR_SERVER_ENUM,
  930. "Non local domain %ws - Check for double hop\n",
  931. Domain));
  932. if ( Network->Role & ROLE_MASTER ) {
  933. if ( Network->DomainList.EntriesRead != 0 ) {
  934. DomainEntry = LookupInterimServerList(&Network->DomainList, (LPWSTR) Domain);
  935. if (DomainEntry != NULL) {
  936. MasterName = DomainEntry->Comment;
  937. }
  938. } else {
  939. ULONG i;
  940. PSERVER_INFO_101 DomainInfo;
  941. DWORD DoubleHopEntriesRead = 0;
  942. DWORD DoubleHopTotalEntries = 0;
  943. status = BrGetLocalBrowseList(Network,
  944. NULL,
  945. 101,
  946. SV_TYPE_DOMAIN_ENUM,
  947. &DoubleHopLocalList,
  948. &DoubleHopEntriesRead,
  949. &DoubleHopTotalEntries);
  950. for (i = 0 , DomainInfo = DoubleHopLocalList ;
  951. i < DoubleHopEntriesRead ;
  952. i += 1) {
  953. if (!_wcsicmp(Domain, DomainInfo->sv101_name)) {
  954. MasterName = DomainInfo->sv101_comment;
  955. break;
  956. }
  957. DomainInfo += 1;
  958. }
  959. }
  960. } else {
  961. ULONG i;
  962. PSERVER_INFO_101 DomainInfo;
  963. //
  964. // We're running on a backup browser. We want to find the
  965. // name of the master browser by looking in the backup
  966. // server list for this network.
  967. //
  968. for (i = 0 , DomainInfo = Network->BackupDomainList ;
  969. i < Network->TotalBackupDomainListEntries ;
  970. i += 1) {
  971. if (!_wcsicmp(Domain, DomainInfo->sv101_name)) {
  972. MasterName = DomainInfo->sv101_comment;
  973. break;
  974. }
  975. DomainInfo += 1;
  976. }
  977. }
  978. //
  979. // If we couldn't find a master name, bail out right now.
  980. //
  981. if (MasterName == NULL || *MasterName == UNICODE_NULL) {
  982. try_return(status = ERROR_NO_BROWSER_SERVERS_FOUND);
  983. }
  984. //
  985. // If the master for this domain isn't listed as our
  986. // current machine, remote the API to that server.
  987. //
  988. if (STRICMP(MasterName, Network->DomainInfo->DomUnicodeComputerName)) {
  989. WCHAR RemoteComputerName[UNLEN+1];
  990. //
  991. // Build the name of the remote computer.
  992. //
  993. STRCPY(RemoteComputerName, TEXT("\\\\"));
  994. STRCAT(RemoteComputerName, MasterName);
  995. ASSERT (NetworkLocked);
  996. UNLOCK_NETWORK(Network);
  997. NetworkLocked = FALSE;
  998. BrPrint(( BR_SERVER_ENUM,
  999. "Double hop to %ws on %ws\n",
  1000. RemoteComputerName,
  1001. Network->NetworkName.Buffer));
  1002. status = RxNetServerEnum(RemoteComputerName,
  1003. Network->NetworkName.Buffer,
  1004. Level,
  1005. (LPBYTE *)Buffer,
  1006. PreferedMaximumLength,
  1007. EntriesRead,
  1008. TotalEntries,
  1009. ServerType,
  1010. Domain,
  1011. FirstNameToReturn );
  1012. BrPrint(( BR_SERVER_ENUM, "Double hop done\n"));
  1013. if (!LOCK_NETWORK_SHARED (Network)) {
  1014. try_return(status = NERR_InternalError);
  1015. }
  1016. NetworkLocked = TRUE;
  1017. try_return(status);
  1018. }
  1019. }
  1020. ASSERT (NetworkLocked);
  1021. if (!ARGUMENT_PRESENT(Domain)) {
  1022. STRCPY(DomainName, Network->DomainInfo->DomUnicodeDomainName);
  1023. }
  1024. //
  1025. // If we are running on the master browser, we want to retrieve the
  1026. // local list, merge it with our interim server list, and
  1027. // return that to the user.
  1028. //
  1029. if (Network->Role & ROLE_MASTER) {
  1030. status = BrRetrieveServerListForMaster(Network,
  1031. Buffer,
  1032. EntriesRead,
  1033. TotalEntries,
  1034. Level,
  1035. ServerType,
  1036. PreferedMaximumLength,
  1037. LocalListOnly,
  1038. (LPWSTR) ClientName,
  1039. DomainName,
  1040. FirstNameToReturn );
  1041. } else {
  1042. status = BrRetrieveServerListForBackup(Network,
  1043. Buffer,
  1044. EntriesRead,
  1045. TotalEntries,
  1046. Level,
  1047. ServerType,
  1048. PreferedMaximumLength,
  1049. FirstNameToReturn );
  1050. }
  1051. try_return(status);
  1052. try_exit:NOTHING;
  1053. } finally {
  1054. #if DBG
  1055. if (status == NERR_Success || status == ERROR_MORE_DATA) {
  1056. BrPrint(( BR_SERVER_ENUM,
  1057. "%ws: %ws: Returning Browse list for %lx with %ld entries for %ws\n",
  1058. Domain,
  1059. Network->NetworkName.Buffer,
  1060. ServerType,
  1061. *EntriesRead,
  1062. ClientName));
  1063. } else {
  1064. BrPrint(( BR_CRITICAL,
  1065. "%ws: %ws: Failing I_BrowserServerEnum: %ld for client %ws\n",
  1066. Domain,
  1067. Network->NetworkName.Buffer,
  1068. status, ClientName));
  1069. }
  1070. #endif
  1071. //
  1072. // If we used the local driver's list to retrieve the list of
  1073. // domains, free up the buffer.
  1074. //
  1075. if (DoubleHopLocalList != NULL) {
  1076. MIDL_user_free(DoubleHopLocalList);
  1077. }
  1078. ASSERT (NetworkLocked);
  1079. }
  1080. return status;
  1081. }
  1082. VOID
  1083. TrimServerList(
  1084. IN DWORD Level,
  1085. IN OUT LPBYTE *Buffer,
  1086. IN OUT LPDWORD EntriesRead,
  1087. IN OUT LPDWORD TotalEntries,
  1088. IN LPCWSTR FirstNameToReturn
  1089. )
  1090. /*++
  1091. Routine Description:
  1092. This routine trims any name from Buffer that's less than FirstNameToReturn.
  1093. The routine is implemented by moving the fixed part of the kept entries to the beginning
  1094. of the allocated buffer. Thay way, the pointer contained in the entries need not be
  1095. relocated and the original buffer can still be used.
  1096. Arguments:
  1097. Level - Level of information in the buffer.
  1098. Buffer - Pointer to address of buffer to trim.
  1099. Returns null (and deallocates the buffer) if all the entries are trimmed.
  1100. EntriesRead - Pointer to number of entries in the buffer.
  1101. Returns the number of entries remaining in the buffer after triming.
  1102. TotalEntries - Pointer to number of entries available from server.
  1103. Returns a number reduced by the number of trimmed entries.
  1104. FirstNameToReturn - Supplies the name of the first server or domain to return
  1105. to the caller. If NULL, enumeration begins with the very first entry.
  1106. Passed name must be the canonical form of the name.
  1107. Return Value:
  1108. Returns the error code for the operation.
  1109. --*/
  1110. {
  1111. DWORD EntryCount;
  1112. DWORD EntryNumber;
  1113. DWORD FixedSize;
  1114. LPBYTE CurrentEntry;
  1115. //
  1116. // Early out if there's nothing to do.
  1117. //
  1118. if ( FirstNameToReturn == NULL || *FirstNameToReturn == L'\0' || *EntriesRead == 0 ) {
  1119. return;
  1120. }
  1121. //
  1122. // Compute the size of each entry.
  1123. //
  1124. switch (Level) {
  1125. case 100:
  1126. FixedSize = sizeof(SERVER_INFO_100);
  1127. break;
  1128. case 101:
  1129. FixedSize = sizeof(SERVER_INFO_101);
  1130. break;
  1131. default:
  1132. NetpAssert( FALSE );
  1133. return;
  1134. }
  1135. //
  1136. // Finding the first entry to return
  1137. //
  1138. EntryCount = *EntriesRead;
  1139. for ( EntryNumber=0; EntryNumber< EntryCount; EntryNumber++ ) {
  1140. LPSERVER_INFO_100 ServerEntry =
  1141. (LPSERVER_INFO_100)( *Buffer + FixedSize * EntryNumber);
  1142. //
  1143. // Found the first entry to return.
  1144. //
  1145. if ( STRCMP( ServerEntry->sv100_name, FirstNameToReturn ) >= 0 ) {
  1146. //
  1147. // If we're returning the whole list,
  1148. // simply return.
  1149. //
  1150. if ( ServerEntry == (LPSERVER_INFO_100)(*Buffer) ) {
  1151. return;
  1152. }
  1153. //
  1154. // Copy the remaining entries to the beginning of the buffer.
  1155. // (Yes, this is an overlapping copy)
  1156. //
  1157. RtlMoveMemory( *Buffer, ServerEntry, (*EntriesRead) * FixedSize );
  1158. return;
  1159. }
  1160. //
  1161. // Account for skipped entries.
  1162. //
  1163. *EntriesRead -= 1;
  1164. *TotalEntries -= 1;
  1165. }
  1166. //
  1167. // If no entries should be returned,
  1168. // deallocate the buffer.
  1169. //
  1170. NetApiBufferFree( *Buffer );
  1171. *Buffer = NULL;
  1172. ASSERT ( *EntriesRead == 0 );
  1173. return;
  1174. } // TrimServerList
  1175. //
  1176. // Context for building a local list of all the Wannish transports.
  1177. //
  1178. typedef struct _BR_LOCAL_LIST_CONTEXT {
  1179. LPWSTR DomainName;
  1180. DWORD Level;
  1181. DWORD ServerType;
  1182. PVOID Buffer;
  1183. NET_API_STATUS NetStatus;
  1184. DWORD EntriesRead;
  1185. DWORD TotalEntries;
  1186. BOOLEAN AtLeastOneWorked;
  1187. INTERIM_SERVER_LIST InterimServerList;
  1188. } BR_LOCAL_LIST_CONTEXT, *PBR_LOCAL_LIST_CONTEXT;
  1189. NET_API_STATUS
  1190. BrGetWannishLocalList(
  1191. IN PNETWORK Network,
  1192. IN PVOID Context
  1193. )
  1194. /*++
  1195. Routine Description:
  1196. Worker function to get the local list for all WANNISH networks and
  1197. merge them together.
  1198. Arguments:
  1199. Network - The current network to do.
  1200. Context - Context that the accumulated list is built into.
  1201. Return Value:
  1202. NET_API_STATUS - NERR_Success or reason for failure.
  1203. --*/
  1204. {
  1205. NET_API_STATUS NetStatus;
  1206. PBR_LOCAL_LIST_CONTEXT LocalListContext = (PBR_LOCAL_LIST_CONTEXT) Context;
  1207. PVOID Buffer = NULL;
  1208. DWORD EntriesRead;
  1209. DWORD TotalEntries;
  1210. //
  1211. // If this isn't a wannish network,
  1212. // ignore it.
  1213. if ((Network->Flags & NETWORK_WANNISH) == 0 ) {
  1214. BrPrint(( BR_SERVER_ENUM,
  1215. "%ws: %ws: isn't a wannish network. (ignored)\n",
  1216. Network->DomainInfo->DomUnicodeDomainName,
  1217. Network->NetworkName.Buffer ));
  1218. return NO_ERROR;
  1219. }
  1220. //
  1221. // Get the local list from this transport.
  1222. //
  1223. BrPrint(( BR_SERVER_ENUM,
  1224. "%ws: %ws: Get local list from wannish network.\n",
  1225. Network->DomainInfo->DomUnicodeDomainName,
  1226. Network->NetworkName.Buffer ));
  1227. NetStatus = BrGetLocalBrowseList(
  1228. Network,
  1229. LocalListContext->DomainName,
  1230. LocalListContext->Level,
  1231. LocalListContext->ServerType,
  1232. &Buffer,
  1233. &EntriesRead,
  1234. &TotalEntries);
  1235. if ( NetStatus != NO_ERROR && NetStatus != ERROR_MORE_DATA ) {
  1236. BrPrint(( BR_CRITICAL,
  1237. "%ws: %ws: Get local list from wannish network failed: %ld.\n",
  1238. Network->DomainInfo->DomUnicodeDomainName,
  1239. Network->NetworkName.Buffer,
  1240. NetStatus ));
  1241. //
  1242. // If no transport has worked yet, save this as the new default status.
  1243. // (But keep on going)
  1244. //
  1245. if ( !LocalListContext->AtLeastOneWorked ) {
  1246. LocalListContext->NetStatus = NetStatus;
  1247. }
  1248. NetStatus = NO_ERROR;
  1249. goto Cleanup;
  1250. }
  1251. LocalListContext->AtLeastOneWorked = TRUE;
  1252. LocalListContext->NetStatus = NetStatus;
  1253. //
  1254. // If no data was returned,
  1255. // we're done.
  1256. //
  1257. if ( EntriesRead == 0 ) {
  1258. BrPrint(( BR_SERVER_ENUM,
  1259. "%ws: %ws: Get local list from wannish network got zero entries.\n",
  1260. Network->DomainInfo->DomUnicodeDomainName,
  1261. Network->NetworkName.Buffer ));
  1262. NetStatus = NO_ERROR;
  1263. goto Cleanup;
  1264. }
  1265. //
  1266. // If a prior network already returned some entries,
  1267. // build an interim list to merge both lists into.
  1268. //
  1269. if ( LocalListContext->Buffer != NULL ) {
  1270. NetStatus = MergeServerList(
  1271. &LocalListContext->InterimServerList,
  1272. LocalListContext->Level,
  1273. LocalListContext->Buffer,
  1274. LocalListContext->EntriesRead,
  1275. LocalListContext->TotalEntries );
  1276. MIDL_user_free(LocalListContext->Buffer);
  1277. LocalListContext->Buffer = NULL;
  1278. LocalListContext->EntriesRead = 0;
  1279. LocalListContext->TotalEntries = 0;;
  1280. if ( NetStatus != NO_ERROR ) {
  1281. BrPrint(( BR_CRITICAL,
  1282. "%ws: %ws: merge local list from wannish network failed: %ld.\n",
  1283. Network->DomainInfo->DomUnicodeDomainName,
  1284. Network->NetworkName.Buffer,
  1285. NetStatus ));
  1286. goto Cleanup;
  1287. }
  1288. }
  1289. //
  1290. // If there is an interim server list,
  1291. // add the new entries to it.
  1292. //
  1293. if ( LocalListContext->InterimServerList.TotalEntries != 0 ) {
  1294. NetStatus = MergeServerList(
  1295. &LocalListContext->InterimServerList,
  1296. LocalListContext->Level,
  1297. Buffer,
  1298. EntriesRead,
  1299. TotalEntries );
  1300. if ( NetStatus != NO_ERROR ) {
  1301. BrPrint(( BR_CRITICAL,
  1302. "%ws: %ws: merge local list from wannish network failed: %ld.\n",
  1303. Network->DomainInfo->DomUnicodeDomainName,
  1304. Network->NetworkName.Buffer,
  1305. NetStatus ));
  1306. goto Cleanup;
  1307. }
  1308. //
  1309. // If this is the first network to return entries,
  1310. // simply save the list.
  1311. //
  1312. // This saves us from having to convert a list to an interim list just
  1313. // to have to convert it back in those cases where we have a single wannish
  1314. // network.
  1315. //
  1316. } else {
  1317. LocalListContext->Buffer = Buffer;
  1318. LocalListContext->EntriesRead = EntriesRead;
  1319. LocalListContext->TotalEntries = TotalEntries;
  1320. Buffer = NULL;
  1321. }
  1322. NetStatus = NO_ERROR;
  1323. Cleanup:
  1324. if ( Buffer != NULL ) {
  1325. MIDL_user_free(Buffer);
  1326. }
  1327. return NetStatus;
  1328. }
  1329. NET_API_STATUS
  1330. BrRetrieveServerListForMaster(
  1331. IN PNETWORK Network,
  1332. IN OUT PVOID *Buffer,
  1333. OUT PDWORD EntriesRead,
  1334. OUT PDWORD TotalEntries,
  1335. IN DWORD Level,
  1336. IN DWORD ServerType,
  1337. IN DWORD PreferedMaximumLength,
  1338. IN BOOLEAN LocalListOnly,
  1339. IN LPTSTR ClientName,
  1340. IN LPTSTR DomainName,
  1341. IN LPCWSTR FirstNameToReturn
  1342. )
  1343. {
  1344. BOOLEAN GetLocalList = FALSE;
  1345. NET_API_STATUS status;
  1346. BOOLEAN EarlyOut = FALSE;
  1347. //
  1348. // Determine if it is appropriate that we should early out after retrieving
  1349. // the list of servers (or domains) from the bowser.
  1350. //
  1351. // We will early out on the following critieria:
  1352. //
  1353. // 1) If we are requested to retrieve the local list on a Wannish xport
  1354. // 2) If the transport isn't a wannish transport, or
  1355. // 3) If this domain isn't our primary domain (in which case it's an
  1356. // "otherdomain" request).
  1357. //
  1358. if (LocalListOnly || STRICMP(DomainName, Network->DomainInfo->DomUnicodeDomainName)) {
  1359. EarlyOut = TRUE;
  1360. } else if (!(Network->Flags & NETWORK_WANNISH)) {
  1361. //
  1362. // This isn't a wannish transport. We need to see if we've been
  1363. // master long enough for the driver to have retrieved a reasonable
  1364. // list.
  1365. //
  1366. // The server and domain table will be emptied when the first master
  1367. // browser timer is run. This will happen 15 minutes after we
  1368. // become the master, so we will use our services list for the
  1369. // first 15 minutes the browser is master.
  1370. //
  1371. if (ServerType == SV_TYPE_DOMAIN_ENUM) {
  1372. if (Network->DomainList.EntriesRead == 0) {
  1373. EarlyOut = TRUE;
  1374. }
  1375. } else {
  1376. if (Network->BrowseTable.EntriesRead == 0) {
  1377. EarlyOut = TRUE;
  1378. }
  1379. }
  1380. }
  1381. if (EarlyOut) {
  1382. GetLocalList = TRUE;
  1383. } else if (ServerType == SV_TYPE_ALL) {
  1384. if ((BrCurrentSystemTime() - Network->LastBowserServerQueried) > BrInfo.DriverQueryFrequency ) {
  1385. GetLocalList = TRUE;
  1386. }
  1387. } else if (ServerType == SV_TYPE_DOMAIN_ENUM) {
  1388. if ((BrCurrentSystemTime() - Network->LastBowserDomainQueried) > BrInfo.DriverQueryFrequency ) {
  1389. GetLocalList = TRUE;
  1390. }
  1391. } else {
  1392. GetLocalList = TRUE;
  1393. }
  1394. if (GetLocalList && !EarlyOut) {
  1395. //
  1396. // If we're retriving the list from the driver, and can't early out
  1397. // this request, this means that we will be merging this server list
  1398. // with an interim server list. This means that we will be modifying
  1399. // the network structure, and thus that we need to re-lock the network
  1400. // structure.
  1401. //
  1402. UNLOCK_NETWORK(Network);
  1403. if (!LOCK_NETWORK(Network)) {
  1404. return NERR_InternalError;
  1405. }
  1406. //
  1407. // Now re-test to see if another thread came in and retrieved the
  1408. // list from the driver while we had the network unlocked. If so,
  1409. // we don't want to get the local list any more.
  1410. //
  1411. // Otherwise, we want to update the last query time.
  1412. //
  1413. if (ServerType == SV_TYPE_ALL) {
  1414. if ((BrCurrentSystemTime() - Network->LastBowserServerQueried) > BrInfo.DriverQueryFrequency ) {
  1415. Network->LastBowserServerQueried = BrCurrentSystemTime();
  1416. } else {
  1417. GetLocalList = FALSE;
  1418. }
  1419. } else if (ServerType == SV_TYPE_DOMAIN_ENUM) {
  1420. if ((BrCurrentSystemTime() - Network->LastBowserDomainQueried) > BrInfo.DriverQueryFrequency ) {
  1421. Network->LastBowserDomainQueried = BrCurrentSystemTime();
  1422. } else {
  1423. GetLocalList = FALSE;
  1424. }
  1425. } else {
  1426. Network->LastBowserServerQueried = BrCurrentSystemTime();
  1427. }
  1428. }
  1429. //
  1430. // If we're supposed to retrieve the local server list, retrieve it.
  1431. //
  1432. if (GetLocalList) {
  1433. DWORD ServerTypeForLocalList;
  1434. //
  1435. // Calculate the server type to use when retrieving the list from
  1436. // the driver.
  1437. //
  1438. if (LocalListOnly ||
  1439. (ServerType == SV_TYPE_DOMAIN_ENUM) ||
  1440. !(Network->Flags & NETWORK_WANNISH)) {
  1441. //
  1442. // If we are retrieving the local list, or this is a non-wannish
  1443. // transport, or we are asking for domains, we want to
  1444. // keep the callers server type.
  1445. //
  1446. ServerTypeForLocalList = ServerType;
  1447. } else {
  1448. //
  1449. // This must be a wannish transport, ask for all servers (we know
  1450. // that we're not asking for domains, since we checked that
  1451. // above).
  1452. //
  1453. // We ask for all the servers because it's possible that a servers
  1454. // role has changed.
  1455. //
  1456. ASSERT (Network->Flags & NETWORK_WANNISH);
  1457. ServerTypeForLocalList = SV_TYPE_ALL;
  1458. }
  1459. BrPrint(( BR_SERVER_ENUM,
  1460. "%ws: %ws: Get local browse list for %ws\n",
  1461. Network->DomainInfo->DomUnicodeDomainName,
  1462. Network->NetworkName.Buffer,
  1463. ClientName));
  1464. //
  1465. // If this is a call from the Domain Master Browser to get the local list,
  1466. // and this is a wannish transport,
  1467. // get the local list on all wannish transports.
  1468. //
  1469. if ( LocalListOnly &&
  1470. (Network->Flags & NETWORK_WANNISH) != 0 ) {
  1471. BR_LOCAL_LIST_CONTEXT LocalListContext;
  1472. //
  1473. // Build a context to use to gather the server lists from all
  1474. // wannish transports
  1475. //
  1476. LocalListContext.DomainName = DomainName;
  1477. LocalListContext.Level = Level;
  1478. LocalListContext.ServerType = ServerTypeForLocalList;
  1479. LocalListContext.Buffer = NULL;
  1480. LocalListContext.NetStatus = NO_ERROR;
  1481. LocalListContext.EntriesRead = 0;
  1482. LocalListContext.TotalEntries = 0;
  1483. LocalListContext.AtLeastOneWorked = FALSE;
  1484. (VOID) InitializeInterimServerList(&LocalListContext.InterimServerList, NULL, NULL, NULL, NULL);
  1485. //
  1486. // Gather the lists
  1487. //
  1488. status = BrEnumerateNetworks( BrGetWannishLocalList, &LocalListContext );
  1489. if ( status == NO_ERROR ) {
  1490. //
  1491. // If multiple networks were found,
  1492. // pack the interim list.
  1493. //
  1494. if ( LocalListContext.InterimServerList.TotalEntries != 0 ) {
  1495. status = PackServerList( &LocalListContext.InterimServerList,
  1496. LocalListContext.Level,
  1497. LocalListContext.ServerType,
  1498. PreferedMaximumLength,
  1499. Buffer,
  1500. EntriesRead,
  1501. TotalEntries,
  1502. (LPWSTR)FirstNameToReturn );
  1503. //
  1504. // If just a single network was found,
  1505. // just use it.
  1506. //
  1507. } else {
  1508. status = LocalListContext.NetStatus;
  1509. *Buffer = LocalListContext.Buffer;
  1510. *EntriesRead = LocalListContext.EntriesRead;
  1511. *TotalEntries = LocalListContext.TotalEntries;
  1512. }
  1513. }
  1514. (VOID) UninitializeInterimServerList( &LocalListContext.InterimServerList );
  1515. } else {
  1516. status = BrGetLocalBrowseList(
  1517. Network,
  1518. DomainName,
  1519. ( EarlyOut ? Level : 101 ),
  1520. ServerTypeForLocalList,
  1521. Buffer,
  1522. EntriesRead,
  1523. TotalEntries);
  1524. }
  1525. // BrPrint(( BR_SERVER_ENUM, "List retrieved. %ld entries, %ld total\n", *EntriesRead, *TotalEntries));
  1526. //
  1527. // If we're supposed to early-out this request (or if there was
  1528. // an error), do so now.
  1529. //
  1530. if (EarlyOut ||
  1531. (status != NERR_Success && status != ERROR_MORE_DATA)) {
  1532. //
  1533. // If we're returning an early out list,
  1534. // truncate the complete list returned from the kernel.
  1535. //
  1536. // This saves us from having to modify the kernel interface and untangle
  1537. // the code above.
  1538. //
  1539. if ( status == NERR_Success || status == ERROR_MORE_DATA ) {
  1540. TrimServerList( Level,
  1541. (LPBYTE *)Buffer,
  1542. EntriesRead,
  1543. TotalEntries,
  1544. FirstNameToReturn );
  1545. }
  1546. BrPrint(( BR_SERVER_ENUM, "Early out for %ws with %ld servers. Don't merge server list.\n", ClientName, *EntriesRead));
  1547. return status;
  1548. }
  1549. if (status == NERR_Success || status == ERROR_MORE_DATA) {
  1550. if (*EntriesRead != 0) {
  1551. //
  1552. // Merge the local list with the list we got from the
  1553. // master or from the domain master.
  1554. //
  1555. BrPrint(( BR_SERVER_ENUM,
  1556. "%ws: %ws: Merge %d entries in server list for %ws \n",
  1557. Network->DomainInfo->DomUnicodeDomainName,
  1558. Network->NetworkName.Buffer,
  1559. *EntriesRead,
  1560. ClientName));
  1561. status = MergeServerList((ServerType == SV_TYPE_DOMAIN_ENUM ?
  1562. &Network->DomainList :
  1563. &Network->BrowseTable),
  1564. 101,
  1565. *Buffer,
  1566. *EntriesRead,
  1567. *TotalEntries
  1568. );
  1569. }
  1570. }
  1571. }
  1572. //
  1573. // We've merged the local list into the appropriate interim table,
  1574. // now free it up.
  1575. //
  1576. if (*EntriesRead != 0) {
  1577. MIDL_user_free(*Buffer);
  1578. }
  1579. status = PackServerList((ServerType == SV_TYPE_DOMAIN_ENUM ?
  1580. &Network->DomainList :
  1581. &Network->BrowseTable),
  1582. Level,
  1583. ServerType,
  1584. PreferedMaximumLength,
  1585. Buffer,
  1586. EntriesRead,
  1587. TotalEntries,
  1588. FirstNameToReturn
  1589. );
  1590. return status;
  1591. }
  1592. NET_API_STATUS
  1593. BrRetrieveServerListForBackup(
  1594. IN PNETWORK Network,
  1595. IN OUT PVOID *Buffer,
  1596. OUT PDWORD EntriesRead,
  1597. OUT PDWORD TotalEntries,
  1598. IN DWORD Level,
  1599. IN DWORD ServerType,
  1600. IN DWORD PreferedMaximumLength,
  1601. IN LPCWSTR FirstNameToReturn
  1602. )
  1603. {
  1604. PSERVER_INFO_101 ServerList, ClientServerInfo;
  1605. ULONG EntriesInList;
  1606. ULONG TotalEntriesInList;
  1607. ULONG EntrySize;
  1608. ULONG BufferSize;
  1609. LPTSTR BufferEnd;
  1610. BOOLEAN ReturnWholeList = FALSE;
  1611. BOOLEAN TrimmingNames;
  1612. BOOLEAN BufferFull = FALSE; // see bug 427656
  1613. //
  1614. // If we are not running as a master, we want to use our stored
  1615. // server list to figure out what the client gets.
  1616. //
  1617. if (ServerType == SV_TYPE_DOMAIN_ENUM) {
  1618. ServerList = Network->BackupDomainList;
  1619. TotalEntriesInList = EntriesInList = Network->TotalBackupDomainListEntries;
  1620. ReturnWholeList = TRUE;
  1621. } else {
  1622. ServerList = Network->BackupServerList;
  1623. TotalEntriesInList = EntriesInList = Network->TotalBackupServerListEntries;
  1624. if (ServerType == SV_TYPE_ALL) {
  1625. ReturnWholeList = TRUE;
  1626. }
  1627. }
  1628. //
  1629. // Figure out the largest buffer we have to allocate to hold this
  1630. // server info.
  1631. //
  1632. if (Level == 101) {
  1633. if (PreferedMaximumLength == MAXULONG) {
  1634. if (ServerType == SV_TYPE_DOMAIN_ENUM) {
  1635. BufferSize = (sizeof(SERVER_INFO_101) + (CNLEN+1 + CNLEN+1)*sizeof(TCHAR)) * EntriesInList;
  1636. } else {
  1637. BufferSize = (sizeof(SERVER_INFO_101) + (CNLEN+1 + LM20_MAXCOMMENTSZ+1)*sizeof(TCHAR)) * EntriesInList;
  1638. }
  1639. } else {
  1640. BufferSize = PreferedMaximumLength;
  1641. }
  1642. EntrySize = sizeof(SERVER_INFO_101);
  1643. } else {
  1644. if (PreferedMaximumLength == MAXULONG) {
  1645. BufferSize = (sizeof(SERVER_INFO_100) + (CNLEN+1)*sizeof(TCHAR)) * EntriesInList;
  1646. } else {
  1647. BufferSize = PreferedMaximumLength;
  1648. }
  1649. EntrySize = sizeof(SERVER_INFO_100);
  1650. }
  1651. *Buffer = MIDL_user_allocate(BufferSize);
  1652. if (*Buffer == NULL) {
  1653. return ERROR_NOT_ENOUGH_MEMORY;
  1654. }
  1655. BufferEnd = (LPTSTR)((ULONG_PTR)*Buffer+BufferSize);
  1656. ClientServerInfo = *Buffer;
  1657. *TotalEntries = 0;
  1658. *EntriesRead = 0;
  1659. //
  1660. // While there are still entries to process....
  1661. //
  1662. TrimmingNames = (FirstNameToReturn != NULL && *FirstNameToReturn != L'\0');
  1663. while (EntriesInList) {
  1664. EntriesInList -= 1;
  1665. //
  1666. // If this entry is appropriate to be packed,
  1667. //
  1668. if ( (ServerList->sv101_type & ServerType) &&
  1669. (!TrimmingNames ||
  1670. STRCMP( ServerList->sv101_name, FirstNameToReturn ) >= 0 ) ) {
  1671. TrimmingNames = FALSE;
  1672. //
  1673. // Indicate one more entry in the list.
  1674. //
  1675. *TotalEntries += 1;
  1676. //
  1677. // If we can fit this entry before the buffer end,
  1678. // pack in the information into the buffer.
  1679. //
  1680. if ( !BufferFull &&
  1681. (ULONG_PTR)ClientServerInfo+EntrySize <= (ULONG_PTR)BufferEnd) {
  1682. //
  1683. // Copy over the platform ID and computer name.
  1684. //
  1685. ClientServerInfo->sv101_platform_id = ServerList->sv101_platform_id;
  1686. ClientServerInfo->sv101_name = ServerList->sv101_name;
  1687. if (NetpPackString(&ClientServerInfo->sv101_name,
  1688. (LPBYTE)((PCHAR)ClientServerInfo)+EntrySize,
  1689. &BufferEnd)) {
  1690. if (Level == 101) {
  1691. ClientServerInfo->sv101_version_major = ServerList->sv101_version_major;
  1692. ClientServerInfo->sv101_version_minor = ServerList->sv101_version_minor;
  1693. ClientServerInfo->sv101_type = ServerList->sv101_type;
  1694. ClientServerInfo->sv101_comment = ServerList->sv101_comment;
  1695. if (NetpPackString(&ClientServerInfo->sv101_comment,
  1696. (LPBYTE)((PCHAR)ClientServerInfo)+EntrySize,
  1697. &BufferEnd)) {
  1698. *EntriesRead += 1;
  1699. }
  1700. else {
  1701. BufferFull = TRUE;
  1702. }
  1703. } else {
  1704. *EntriesRead += 1;
  1705. }
  1706. }
  1707. else {
  1708. BufferFull = TRUE;
  1709. }
  1710. ClientServerInfo = (PSERVER_INFO_101)((PCHAR)ClientServerInfo+EntrySize);
  1711. } else {
  1712. //
  1713. // If we're returning the entire list, we can
  1714. // early out now, since there's no point in continuing.
  1715. //
  1716. if (ReturnWholeList) {
  1717. *TotalEntries = TotalEntriesInList;
  1718. break;
  1719. }
  1720. BufferFull = TRUE;
  1721. }
  1722. }
  1723. ServerList += 1;
  1724. }
  1725. //
  1726. // If we weren't able to pack all the entries into the list,
  1727. // return ERROR_MORE_DATA
  1728. //
  1729. if (*EntriesRead != *TotalEntries) {
  1730. return ERROR_MORE_DATA;
  1731. } else {
  1732. return NERR_Success;
  1733. }
  1734. }
  1735. NET_API_STATUS
  1736. I_BrowserrResetStatistics (
  1737. IN LPTSTR servername OPTIONAL
  1738. )
  1739. {
  1740. NET_API_STATUS Status = NERR_Success;
  1741. ULONG BufferSize;
  1742. //
  1743. // Perform access validation on the caller.
  1744. //
  1745. Status = NetpAccessCheck(
  1746. BrGlobalBrowserSecurityDescriptor, // Security descriptor
  1747. BROWSER_CONTROL_ACCESS, // Desired access
  1748. &BrGlobalBrowserInfoMapping ); // Generic mapping
  1749. if ( Status != NERR_Success) {
  1750. BrPrint((BR_CRITICAL,
  1751. "I_BrowserrResetStatistics failed NetpAccessCheck\n" ));
  1752. return ERROR_ACCESS_DENIED;
  1753. }
  1754. EnterCriticalSection(&BrowserStatisticsLock);
  1755. NumberOfServerEnumerations = 0;
  1756. NumberOfDomainEnumerations = 0;
  1757. NumberOfOtherEnumerations = 0;
  1758. NumberOfMissedGetBrowserListRequests = 0;
  1759. //
  1760. // Reset the driver's statistics as well.
  1761. //
  1762. if (!DeviceIoControl(BrDgReceiverDeviceHandle, IOCTL_LMDR_RESET_STATISTICS, NULL, 0, NULL, 0, &BufferSize, NULL)) {
  1763. //
  1764. // The API failed, return the error.
  1765. //
  1766. Status = GetLastError();
  1767. }
  1768. LeaveCriticalSection(&BrowserStatisticsLock);
  1769. return Status;
  1770. }
  1771. NET_API_STATUS
  1772. I_BrowserrQueryStatistics (
  1773. IN LPTSTR servername OPTIONAL,
  1774. OUT LPBROWSER_STATISTICS *Statistics
  1775. )
  1776. {
  1777. NET_API_STATUS Status = NERR_Success;
  1778. BOWSER_STATISTICS BowserStatistics;
  1779. ULONG BufferSize;
  1780. //
  1781. // Perform access validation on the caller.
  1782. //
  1783. Status = NetpAccessCheck(
  1784. BrGlobalBrowserSecurityDescriptor, // Security descriptor
  1785. BROWSER_QUERY_ACCESS, // Desired access
  1786. &BrGlobalBrowserInfoMapping ); // Generic mapping
  1787. if ( Status != NERR_Success) {
  1788. BrPrint((BR_CRITICAL,
  1789. "I_BrowserrQueryStatistics failed NetpAccessCheck\n" ));
  1790. return ERROR_ACCESS_DENIED;
  1791. }
  1792. *Statistics = MIDL_user_allocate(sizeof(BROWSER_STATISTICS));
  1793. if (*Statistics == NULL) {
  1794. return ERROR_NOT_ENOUGH_MEMORY;
  1795. }
  1796. EnterCriticalSection(&BrowserStatisticsLock);
  1797. if (!DeviceIoControl(BrDgReceiverDeviceHandle, IOCTL_LMDR_QUERY_STATISTICS, NULL, 0, &BowserStatistics, sizeof(BowserStatistics), &BufferSize, NULL)) {
  1798. //
  1799. // The API failed, return the error.
  1800. //
  1801. Status = GetLastError();
  1802. } else {
  1803. if (BufferSize != sizeof(BOWSER_STATISTICS)) {
  1804. Status = ERROR_INSUFFICIENT_BUFFER;
  1805. } else {
  1806. (*Statistics)->StatisticsStartTime = BowserStatistics.StartTime;
  1807. (*Statistics)->NumberOfServerAnnouncements = BowserStatistics.NumberOfServerAnnouncements;
  1808. (*Statistics)->NumberOfDomainAnnouncements = BowserStatistics.NumberOfDomainAnnouncements;
  1809. (*Statistics)->NumberOfElectionPackets = BowserStatistics.NumberOfElectionPackets;
  1810. (*Statistics)->NumberOfMailslotWrites = BowserStatistics.NumberOfMailslotWrites;
  1811. (*Statistics)->NumberOfGetBrowserServerListRequests = BowserStatistics.NumberOfGetBrowserServerListRequests;
  1812. (*Statistics)->NumberOfMissedServerAnnouncements = BowserStatistics.NumberOfMissedServerAnnouncements;
  1813. (*Statistics)->NumberOfMissedMailslotDatagrams = BowserStatistics.NumberOfMissedMailslotDatagrams;
  1814. (*Statistics)->NumberOfMissedGetBrowserServerListRequests = BowserStatistics.NumberOfMissedGetBrowserServerListRequests +
  1815. NumberOfMissedGetBrowserListRequests;
  1816. (*Statistics)->NumberOfFailedServerAnnounceAllocations = BowserStatistics.NumberOfFailedServerAnnounceAllocations;
  1817. (*Statistics)->NumberOfFailedMailslotAllocations = BowserStatistics.NumberOfFailedMailslotAllocations;
  1818. (*Statistics)->NumberOfFailedMailslotReceives = BowserStatistics.NumberOfFailedMailslotReceives;
  1819. (*Statistics)->NumberOfFailedMailslotWrites = BowserStatistics.NumberOfFailedMailslotWrites;
  1820. (*Statistics)->NumberOfFailedMailslotOpens = BowserStatistics.NumberOfFailedMailslotOpens;
  1821. (*Statistics)->NumberOfDuplicateMasterAnnouncements = BowserStatistics.NumberOfDuplicateMasterAnnouncements;
  1822. (*Statistics)->NumberOfIllegalDatagrams = BowserStatistics.NumberOfIllegalDatagrams;
  1823. //
  1824. // Now fill in the local statistics.
  1825. //
  1826. (*Statistics)->NumberOfServerEnumerations = NumberOfServerEnumerations;
  1827. (*Statistics)->NumberOfDomainEnumerations = NumberOfDomainEnumerations;
  1828. (*Statistics)->NumberOfOtherEnumerations = NumberOfOtherEnumerations;
  1829. }
  1830. }
  1831. LeaveCriticalSection(&BrowserStatisticsLock);
  1832. return Status;
  1833. }
  1834. //
  1835. // Browser request response cache management logic.
  1836. //
  1837. PCACHED_BROWSE_RESPONSE
  1838. BrLookupAndAllocateCachedEntry(
  1839. IN PNETWORK Network,
  1840. IN DWORD ServerType,
  1841. IN WORD Size,
  1842. IN DWORD Level,
  1843. IN LPCWSTR FirstNameToReturn
  1844. )
  1845. /*++
  1846. Routine Description:
  1847. This function will look up (and allocate if appropriate) a cached
  1848. browse response for this browse.
  1849. Enter with the Network Locked shared or exclusive.
  1850. Arguments:
  1851. IN PNETWORK Network - Network to allocate entry on.
  1852. IN DWORD ServerType - Server type bits for request.
  1853. IN WORD Size, - Users buffer size for request.
  1854. IN WORD Level - Level of request.
  1855. FirstNameToReturn - Supplies the name of the first domain or server entry to return.
  1856. The caller can use this parameter to implement a resume handle of sorts by passing
  1857. the name of the last entry returned on a previous call. (Notice that the specified
  1858. entry will, also, be returned on this call unless it has since been deleted.)
  1859. Passed name must be the canonical form of the name.
  1860. This entry is never NULL. It may be a pointer to an empty string to indicate
  1861. the enumeration starts at the beginning of the list.
  1862. Return Value:
  1863. PCACHED_BROWSE_RESPONSE - NULL or a cached response for the request.
  1864. --*/
  1865. {
  1866. PLIST_ENTRY entry;
  1867. PCACHED_BROWSE_RESPONSE response;
  1868. //
  1869. // If we have more cached responses than we are allowed,
  1870. // remove the last entry from the list and free it.
  1871. //
  1872. if (Network->NumberOfCachedResponses > BrInfo.NumberOfCachedResponses) {
  1873. //
  1874. // We need to release the network and re-acquire it
  1875. // exclusively, because we use the network lock to protect
  1876. // enumerations from deletions.
  1877. //
  1878. UNLOCK_NETWORK(Network);
  1879. if (LOCK_NETWORK(Network)) {
  1880. EnterCriticalSection(&Network->ResponseCacheLock);
  1881. if (Network->NumberOfCachedResponses > BrInfo.NumberOfCachedResponses) {
  1882. PLIST_ENTRY LastEntry = RemoveTailList(&Network->ResponseCache);
  1883. response = CONTAINING_RECORD(LastEntry, CACHED_BROWSE_RESPONSE, Next);
  1884. Network->NumberOfCachedResponses -= 1;
  1885. response->Next.Flink = NULL;
  1886. response->Next.Blink = NULL;
  1887. //
  1888. // Free the last cached entry.
  1889. //
  1890. BrDestroyCacheEntry( response );
  1891. response = NULL;
  1892. }
  1893. LeaveCriticalSection(&Network->ResponseCacheLock);
  1894. }
  1895. }
  1896. //
  1897. // Search the list of responses for this one.
  1898. //
  1899. EnterCriticalSection(&Network->ResponseCacheLock);
  1900. for (entry = Network->ResponseCache.Flink ;
  1901. entry != &Network->ResponseCache ;
  1902. entry = entry->Flink ) {
  1903. response = CONTAINING_RECORD(entry, CACHED_BROWSE_RESPONSE, Next);
  1904. //
  1905. // If this response cache entry matches the incoming request,
  1906. // we can increment the hit count for this entry and return it.
  1907. //
  1908. if (response->Level == Level
  1909. &&
  1910. response->ServerType == ServerType
  1911. &&
  1912. response->Size == Size
  1913. &&
  1914. wcscmp( response->FirstNameToReturn, FirstNameToReturn ) == 0) {
  1915. //
  1916. // This response exactly matches the request.
  1917. //
  1918. // Bump its hit count and move it to the head of the cache.
  1919. //
  1920. response->HitCount += 1;
  1921. response->TotalHitCount += 1;
  1922. //
  1923. // Remove this entry from its current location in the list and
  1924. // move it to the head of the list.
  1925. //
  1926. RemoveEntryList(&response->Next);
  1927. InsertHeadList(&Network->ResponseCache, &response->Next);
  1928. BrPrint(( BR_SERVER_ENUM,
  1929. "%ws: %ws: Found cache entry 0x%x/%d/%x H:%d T:%d\n",
  1930. Network->DomainInfo->DomUnicodeDomainName,
  1931. Network->NetworkName.Buffer,
  1932. response->ServerType,
  1933. response->Level,
  1934. response->Size,
  1935. response->HitCount,
  1936. response->TotalHitCount ));
  1937. LeaveCriticalSection(&Network->ResponseCacheLock);
  1938. return response;
  1939. }
  1940. }
  1941. //
  1942. // We've walked our entire cache and have been unable to find
  1943. // a response that matches our request.
  1944. //
  1945. // Allocate a new response cache entry and hook it into the cache.
  1946. //
  1947. response = BrAllocateResponseCacheEntry(Network, ServerType, Size, Level, FirstNameToReturn );
  1948. LeaveCriticalSection(&Network->ResponseCacheLock);
  1949. return response;
  1950. }
  1951. VOID
  1952. BrAgeResponseCache(
  1953. IN PNETWORK Network
  1954. )
  1955. /*++
  1956. Routine Description:
  1957. This function will age response cache entries for a network.
  1958. We scan the response cache, and every entry that has a cached response
  1959. will be tossed. In addition, any entry that has had less than the
  1960. cache hit limit number of hits since the past scan will also be removed.
  1961. Arguments:
  1962. IN PNETWORK Network - Network to age entries on.
  1963. Return Value:
  1964. None.
  1965. --*/
  1966. {
  1967. PLIST_ENTRY entry;
  1968. EnterCriticalSection(&Network->ResponseCacheLock);
  1969. try {
  1970. for (entry = Network->ResponseCache.Flink ;
  1971. entry != &Network->ResponseCache ;
  1972. entry = entry->Flink ) {
  1973. PCACHED_BROWSE_RESPONSE response = CONTAINING_RECORD(entry, CACHED_BROWSE_RESPONSE, Next);
  1974. //
  1975. // If this response didn't have a hit count high enough during
  1976. // the previous run to justify keeping it around, blow it away.
  1977. //
  1978. if (response->HitCount < BrInfo.CacheHitLimit) {
  1979. response->LowHitCount += 1;
  1980. }
  1981. //
  1982. // If we have CacheHitLimit iterations of low hits, then
  1983. // flush the entry from the cache.
  1984. //
  1985. if (response->LowHitCount > BrInfo.CacheHitLimit) {
  1986. PLIST_ENTRY nextentry = entry->Blink;
  1987. BrPrint(( BR_SERVER_ENUM,
  1988. "%ws: %ws: Flush cache entry for 0x%x/%d/%x H:%d T:%d\n",
  1989. Network->DomainInfo->DomUnicodeDomainName,
  1990. Network->NetworkName.Buffer,
  1991. response->ServerType,
  1992. response->Level,
  1993. response->Size,
  1994. response->HitCount,
  1995. response->TotalHitCount ));
  1996. Network->NumberOfCachedResponses -= 1;
  1997. RemoveEntryList(entry);
  1998. entry->Flink = NULL;
  1999. entry->Blink = NULL;
  2000. BrDestroyCacheEntry(response);
  2001. entry = nextentry;
  2002. //
  2003. // Null out the pointer to make sure we don't use it again.
  2004. //
  2005. response = NULL;
  2006. } else {
  2007. BrPrint(( BR_SERVER_ENUM,
  2008. "%ws: %ws: Retain cache entry 0x%x/%d/%x H:%d T:%d\n",
  2009. Network->DomainInfo->DomUnicodeDomainName,
  2010. Network->NetworkName.Buffer,
  2011. response->ServerType, response->Level, response->Size, response->HitCount, response->TotalHitCount ));
  2012. //
  2013. // We ALWAYS blow away the response buffer during an age pass.
  2014. //
  2015. MIDL_user_free( response->Buffer );
  2016. response->Buffer = NULL;
  2017. //
  2018. // Reset the hit count for this entry for this pass.
  2019. //
  2020. response->HitCount = 0;
  2021. }
  2022. }
  2023. } finally {
  2024. LeaveCriticalSection(&Network->ResponseCacheLock);
  2025. }
  2026. }
  2027. PCACHED_BROWSE_RESPONSE
  2028. BrAllocateResponseCacheEntry(
  2029. IN PNETWORK Network,
  2030. IN DWORD ServerType,
  2031. IN WORD Size,
  2032. IN DWORD Level,
  2033. IN LPCWSTR FirstNameToReturn
  2034. )
  2035. /*++
  2036. Routine Description:
  2037. This function will allocate a new browse response cache entry.
  2038. Arguments:
  2039. IN PNETWORK Network - Network to allocate entry on.
  2040. IN DWORD ServerType - Server type bits for request.
  2041. IN WORD Size, - Users buffer size for request.
  2042. IN WORD Level - Level of request.
  2043. FirstNameToReturn - FirstNameCached
  2044. Return Value:
  2045. PCACHED_BROWSE_RESPONSE - NULL or a cached response for the request.
  2046. NOTE: This is called with the network response cache locked.
  2047. --*/
  2048. {
  2049. PCACHED_BROWSE_RESPONSE response;
  2050. response = MIDL_user_allocate( sizeof( CACHED_BROWSE_RESPONSE ) );
  2051. if ( response == NULL ) {
  2052. return NULL;
  2053. }
  2054. //
  2055. // Flag the information for this response.
  2056. //
  2057. response->ServerType = ServerType;
  2058. response->Size = Size;
  2059. response->Level = Level;
  2060. //
  2061. // Initialize the other fields in the response.
  2062. //
  2063. response->Buffer = NULL;
  2064. response->HitCount = 0;
  2065. response->TotalHitCount = 0;
  2066. response->LowHitCount = 0;
  2067. response->Status = NERR_Success;
  2068. wcscpy( response->FirstNameToReturn, FirstNameToReturn );
  2069. Network->NumberOfCachedResponses += 1;
  2070. //
  2071. // We hook this response into the tail of the cache. We do this
  2072. // because we assume that this request won't be used frequently. If
  2073. // it is, it will move to the head of the cache naturally.
  2074. //
  2075. InsertTailList(&Network->ResponseCache, &response->Next);
  2076. return response;
  2077. }
  2078. NET_API_STATUS
  2079. BrDestroyCacheEntry(
  2080. IN PCACHED_BROWSE_RESPONSE CacheEntry
  2081. )
  2082. /*++
  2083. Routine Description:
  2084. This routine destroys an individual response cache entry.
  2085. Arguments:
  2086. IN PCACHED_BROWSE_RESPONSE CacheEntry - Entry to destroy.
  2087. Return Value:
  2088. NET_API_STATUS - NERR_Success
  2089. --*/
  2090. {
  2091. ASSERT (CacheEntry->Next.Flink == NULL);
  2092. ASSERT (CacheEntry->Next.Blink == NULL);
  2093. if (CacheEntry->Buffer != NULL) {
  2094. MIDL_user_free(CacheEntry->Buffer);
  2095. }
  2096. MIDL_user_free(CacheEntry);
  2097. return NERR_Success;
  2098. }
  2099. NET_API_STATUS
  2100. BrDestroyResponseCache(
  2101. IN PNETWORK Network
  2102. )
  2103. /*++
  2104. Routine Description:
  2105. This routine destroys the entire response cache for a supplied network.
  2106. Arguments:
  2107. IN PNETWORK Network - Network to allocate entry on.
  2108. Return Value:
  2109. NET_API_STATUS - NERR_Success
  2110. --*/
  2111. {
  2112. while (!IsListEmpty(&Network->ResponseCache)) {
  2113. PCACHED_BROWSE_RESPONSE cacheEntry;
  2114. PLIST_ENTRY entry = RemoveHeadList(&Network->ResponseCache);
  2115. entry->Flink = NULL;
  2116. entry->Blink = NULL;
  2117. cacheEntry = CONTAINING_RECORD(entry, CACHED_BROWSE_RESPONSE, Next);
  2118. Network->NumberOfCachedResponses -= 1;
  2119. BrDestroyCacheEntry(cacheEntry);
  2120. }
  2121. ASSERT (Network->NumberOfCachedResponses == 0);
  2122. return NERR_Success;
  2123. }
  2124. NET_API_STATUS
  2125. NetrBrowserStatisticsGet (
  2126. IN LPTSTR servername OPTIONAL,
  2127. IN DWORD Level,
  2128. IN OUT LPBROWSER_STATISTICS_STRUCT InfoStruct
  2129. )
  2130. {
  2131. //
  2132. // And return success.
  2133. //
  2134. return(NERR_Success);
  2135. }
  2136. NET_API_STATUS
  2137. NetrBrowserStatisticsClear (
  2138. IN LPTSTR servername OPTIONAL
  2139. )
  2140. {
  2141. //
  2142. // And return success.
  2143. //
  2144. return(NERR_Success);
  2145. }
  2146. #if DBG
  2147. NET_API_STATUS
  2148. I_BrowserrDebugCall (
  2149. IN LPTSTR servername OPTIONAL,
  2150. IN DWORD DebugCode,
  2151. IN DWORD OptionalValue
  2152. )
  2153. {
  2154. NET_API_STATUS Status = STATUS_SUCCESS;
  2155. //
  2156. // Perform access validation on the caller.
  2157. //
  2158. Status = NetpAccessCheck(
  2159. BrGlobalBrowserSecurityDescriptor, // Security descriptor
  2160. BROWSER_CONTROL_ACCESS, // Desired access
  2161. &BrGlobalBrowserInfoMapping ); // Generic mapping
  2162. if ( Status != NERR_Success) {
  2163. BrPrint((BR_CRITICAL,
  2164. "I_BrowserrDebugCall failed NetpAccessCheck\n" ));
  2165. return ERROR_ACCESS_DENIED;
  2166. }
  2167. switch (DebugCode) {
  2168. case BROWSER_DEBUG_BREAK_POINT:
  2169. DbgBreakPoint();
  2170. break;
  2171. case BROWSER_DEBUG_DUMP_NETWORKS:
  2172. BrDumpNetworks();
  2173. break;
  2174. case BROWSER_DEBUG_SET_DEBUG:
  2175. BrInfo.BrowserDebug |= OptionalValue;
  2176. BrPrint(( BR_INIT, "Setting browser trace to %lx\n", BrInfo.BrowserDebug));
  2177. break;
  2178. case BROWSER_DEBUG_CLEAR_DEBUG:
  2179. BrInfo.BrowserDebug &= ~OptionalValue;
  2180. BrPrint(( BR_INIT, "Setting browser trace to %lx\n", BrInfo.BrowserDebug));
  2181. break;
  2182. case BROWSER_DEBUG_TRUNCATE_LOG:
  2183. Status = BrTruncateLog();
  2184. break;
  2185. default:
  2186. BrPrint(( BR_CRITICAL, "Unknown debug callout %lx\n", DebugCode));
  2187. DbgBreakPoint();
  2188. break;
  2189. }
  2190. return Status;
  2191. }
  2192. NET_API_STATUS
  2193. I_BrowserrDebugTrace (
  2194. IN LPTSTR servername OPTIONAL,
  2195. IN LPSTR String
  2196. )
  2197. {
  2198. NET_API_STATUS Status;
  2199. //
  2200. // Perform access validation on the caller.
  2201. //
  2202. Status = NetpAccessCheck(
  2203. BrGlobalBrowserSecurityDescriptor, // Security descriptor
  2204. BROWSER_CONTROL_ACCESS, // Desired access
  2205. &BrGlobalBrowserInfoMapping ); // Generic mapping
  2206. if ( Status != NERR_Success) {
  2207. BrPrint((BR_CRITICAL,
  2208. "I_BrowserrDebugTrace failed NetpAccessCheck\n" ));
  2209. return ERROR_ACCESS_DENIED;
  2210. }
  2211. //
  2212. // Stick the string parameter into the browser log.
  2213. //
  2214. BrowserTrace( BR_UTIL, "%s", String);
  2215. //
  2216. // And return success.
  2217. //
  2218. return(NERR_Success);
  2219. }
  2220. #else
  2221. NET_API_STATUS
  2222. I_BrowserrDebugCall (
  2223. IN LPTSTR servername OPTIONAL,
  2224. IN DWORD DebugCode,
  2225. IN DWORD OptionalValue
  2226. )
  2227. {
  2228. return(ERROR_NOT_SUPPORTED);
  2229. }
  2230. NET_API_STATUS
  2231. I_BrowserrDebugTrace (
  2232. IN LPTSTR servername OPTIONAL,
  2233. IN LPSTR String
  2234. )
  2235. {
  2236. return(ERROR_NOT_SUPPORTED);
  2237. }
  2238. #endif