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.

1750 lines
40 KiB

  1. /*++
  2. Copyright (c) 1991-1992 Microsoft Corporation
  3. Module Name:
  4. SsInit.c
  5. Abstract:
  6. This module contains initialization routines for the NT server
  7. service.
  8. Author:
  9. David Treadwell (davidtr) 6-Mar-1991
  10. Revision History:
  11. ChuckC 20-May-93 Load share remarks from messagefile so it
  12. can be internationalized.
  13. --*/
  14. #include "srvsvcp.h"
  15. #include "srvconfg.h"
  16. #include "ssreg.h"
  17. #include <netevent.h>
  18. #include <lmapibuf.h> // NetApiBufferFree().
  19. #include <lmconfig.h>
  20. #include <netlib.h>
  21. #include <apperr2.h>
  22. #include <debugfmt.h>
  23. #include <tstr.h>
  24. #define SERVICE_REGISTRY_KEY L"\\Registry\\Machine\\System\\CurrentControlSet\\Services\\"
  25. #define SERVER_DRIVER_NAME L"Srv"
  26. #define MIN(a,b) ( ((a) < (b)) ? (a) : (b) )
  27. #define MAX(a,b) ( ((a) < (b)) ? (b) : (a) )
  28. #define MINIMIZE(_param,_max) _param = MIN( _param, _max );
  29. //
  30. // Internationalizable share remarks.
  31. //
  32. #define NETMSG_DLL TEXT("NETMSG.DLL")
  33. LPWSTR SsDefaultRemark = TEXT("") ; // if all else fails
  34. LPWSTR SsAdminShareRemark = NULL ;
  35. LPWSTR SsIPCShareRemark = NULL ;
  36. LPWSTR SsDiskAdminShareRemark = NULL ;
  37. //
  38. // Lock to protect the ShareDeleteCommit list
  39. //
  40. extern CRITICAL_SECTION ShareDelContextMutex;
  41. extern PSHARE_DEL_CONTEXT SrvShareDelContextHead;
  42. //
  43. // Forward declarations.
  44. //
  45. NET_API_STATUS
  46. CreateDefaultShares (
  47. VOID
  48. );
  49. VOID
  50. InitializeDefaultData(
  51. VOID
  52. );
  53. VOID
  54. InitializeStrings(
  55. VOID
  56. );
  57. VOID
  58. FreeStrings(
  59. VOID
  60. );
  61. NET_API_STATUS
  62. InitializeServer (
  63. VOID
  64. );
  65. NET_API_STATUS
  66. LoadServer (
  67. VOID
  68. );
  69. VOID
  70. SetServerName (
  71. VOID
  72. );
  73. DWORD
  74. DiscoverDrives (
  75. VOID
  76. );
  77. NET_API_STATUS
  78. TerminateServer (
  79. VOID
  80. );
  81. VOID
  82. UnloadServer (
  83. VOID
  84. );
  85. #define IsEmbedded() IsSuiteVersion(VER_SUITE_EMBEDDEDNT)
  86. BOOL IsSuiteVersion(USHORT SuiteMask);
  87. VOID
  88. SAMWaitAnnounce (
  89. LPVOID event
  90. )
  91. {
  92. ULONG i;
  93. //
  94. // Announce ourselves and then wait for awhile.
  95. // If the event gets signaled, terminate the loop and this thread.
  96. // But don't do this forever, since SAM may actually get stuck
  97. //
  98. //
  99. // Do it for 30 minutes
  100. //
  101. for( i=0; i < 120; i++ ) {
  102. AnnounceServiceStatus( 1 );
  103. if( WaitForSingleObject( (HANDLE)event, 15*1000 ) != WAIT_TIMEOUT ) {
  104. break;
  105. }
  106. }
  107. if( i == 120 ) {
  108. DbgPrint( "SRVSVC: SAM has hung on startup. \"Srv\" will be reported as failed to start.\n" );
  109. }
  110. }
  111. NET_API_STATUS
  112. SsInitialize (
  113. IN DWORD argc,
  114. IN LPWSTR argv[]
  115. )
  116. /*++
  117. Routine Description:
  118. This routine controls initialization of the server service and
  119. server driver. It sets up server data stored in the server
  120. service, parses the command line parameters in case any data needs
  121. to be changed, and then starts the file server.
  122. Arguments:
  123. argc - the count of command-line arguments.
  124. argv - an array of pointers to the command line arguments.
  125. Return Value:
  126. NET_API_STATUS - results of operation.
  127. --*/
  128. {
  129. NET_API_STATUS error;
  130. HANDLE threadHandle = NULL;
  131. HANDLE event = NULL;
  132. //
  133. // Initialize the resource that protects access to global server
  134. // information.
  135. //
  136. SS_ASSERT( !SsData.SsServerInfoResourceInitialized );
  137. try {
  138. RtlInitializeResource( &SsData.SsServerInfoResource );
  139. } except( EXCEPTION_EXECUTE_HANDLER ) {
  140. return RtlNtStatusToDosError( GetExceptionCode() );
  141. }
  142. // Initialize the lock that protects the ShareDelCommit list
  143. InitializeCriticalSection( &ShareDelContextMutex );
  144. SrvShareDelContextHead = NULL;
  145. //
  146. // We hold this resource while we are doing announcements, and when
  147. // we communicate with the FSD. These ought to be quick operations,
  148. // but it's really unpredictable under load. Indicate to the RTL
  149. // that we really don't know how long it'll take.
  150. //
  151. SsData.SsServerInfoResource.Flags |= RTL_RESOURCE_FLAG_LONG_TERM;
  152. SsData.SsServerInfoResourceInitialized = TRUE;
  153. //
  154. // Get the internationalizable special share remarks
  155. //
  156. InitializeStrings( );
  157. //
  158. // Initialize the server name list bits list.
  159. //
  160. SsData.SsServerNameList = NULL;
  161. IF_DEBUG(INITIALIZATION) {
  162. SS_PRINT(( "SsInitialize: resource initialized.\n" ));
  163. }
  164. //
  165. // Create the event used for termination synchronization.
  166. //
  167. SS_ASSERT( SsData.SsTerminationEvent == NULL );
  168. SsData.SsTerminationEvent = CreateEvent( NULL, FALSE, FALSE, NULL );
  169. if ( SsData.SsTerminationEvent == NULL ) {
  170. error = GetLastError( );
  171. SS_PRINT(( "SsInitialize: CreateEvent failed: %ld\n", error ));
  172. return error;
  173. }
  174. //
  175. // Initialize the server data to its default values stored in
  176. // srvconfg.h.
  177. //
  178. InitializeDefaultData( );
  179. IF_DEBUG(INITIALIZATION) {
  180. SS_PRINT(( "SsInitialize: default data initialized.\n" ));
  181. }
  182. //
  183. // Sometimes the LSA misbehaves and doesn't process the computer name and
  184. // domain name fetching calls. So we need to defeat the service controller
  185. // hang detection mechanism. Sigh.
  186. //
  187. event = CreateEvent( NULL, TRUE, FALSE, NULL );
  188. if( event != NULL ) {
  189. DWORD threadId;
  190. threadHandle = CreateThread(
  191. NULL,
  192. 0,
  193. (LPTHREAD_START_ROUTINE)SAMWaitAnnounce,
  194. (LPVOID)event,
  195. 0,
  196. &threadId
  197. );
  198. if( threadHandle == NULL ) {
  199. CloseHandle( event );
  200. event = NULL;
  201. }
  202. }
  203. //
  204. // Get the computer name.
  205. //
  206. SetServerName( );
  207. //
  208. // Get the primary domain for this computer into the startup parameters.
  209. //
  210. SsSetDomainName( );
  211. if( event != NULL ) {
  212. //
  213. // We created an announcement thread, set the event telling it to terminate
  214. //
  215. SetEvent( event );
  216. //
  217. // Wait for the thread to terminate
  218. //
  219. (VOID)WaitForSingleObject( threadHandle, INFINITE );
  220. //
  221. // Close the handles
  222. //
  223. CloseHandle( event );
  224. CloseHandle( threadHandle );
  225. }
  226. //
  227. // See if we are the top of a DFS tree
  228. //
  229. SsSetDfsRoot();
  230. //
  231. // Verify that the various registry keys under the main server
  232. // service key exist.
  233. //
  234. error = SsCheckRegistry( );
  235. if ( error != NO_ERROR ) {
  236. IF_DEBUG(INITIALIZATION_ERRORS) {
  237. SS_PRINT(( "SsInitialize: SsCheckRegistry failed: %ld\n", error ));
  238. }
  239. return error;
  240. }
  241. IF_DEBUG(INITIALIZATION) {
  242. SS_PRINT(( "SsInitialize: registry keys verified.\n" ));
  243. }
  244. //
  245. // Load server configuration data from the registry.
  246. //
  247. error = SsLoadConfigurationParameters( );
  248. if ( error != NO_ERROR ) {
  249. IF_DEBUG(INITIALIZATION_ERRORS) {
  250. SS_PRINT(( "SsInitialize: SsLoadConfigurationParameters failed: "
  251. "%ld\n", error ));
  252. }
  253. return error;
  254. }
  255. IF_DEBUG(INITIALIZATION) {
  256. SS_PRINT(( "SsInitialize: configuration parameters loaded.\n" ));
  257. }
  258. //
  259. // Parse the command line. This will change server data values as
  260. // specified.
  261. //
  262. error = SsParseCommandLine( argc, argv, TRUE );
  263. if ( error != NO_ERROR ) {
  264. IF_DEBUG(INITIALIZATION_ERRORS) {
  265. SS_PRINT(( "SsInitialize: SsParseCommandLine failed: %ld\n",
  266. error ));
  267. }
  268. return error;
  269. }
  270. IF_DEBUG(INITIALIZATION) {
  271. SS_PRINT(( "SsInitialize: command line parsed.\n" ));
  272. }
  273. //
  274. // Set up the security objects that will be used to validate access
  275. // for APIs.
  276. //
  277. error = SsCreateSecurityObjects( );
  278. if ( error != NO_ERROR ) {
  279. return error;
  280. }
  281. IF_DEBUG(INITIALIZATION) {
  282. SS_PRINT(( "SsInitialize: security initialized.\n" ));
  283. }
  284. //
  285. // Start the file server driver.
  286. //
  287. error = InitializeServer( );
  288. if ( error != NO_ERROR ) {
  289. return error;
  290. }
  291. IF_DEBUG(INITIALIZATION) {
  292. SS_PRINT(( "SsInitialize: server FSP initialized.\n" ));
  293. }
  294. //
  295. // Watch for Domain Name changes, and automatically pick them up
  296. //
  297. error = NetRegisterDomainNameChangeNotification( &SsData.SsDomainNameChangeEvent );
  298. IF_DEBUG(INITIALIZATION_ERRORS) {
  299. if( error != NO_ERROR ) {
  300. SS_PRINT(( "SsInitialize: NetRegisterDomainNameChangeNotification failed: "
  301. "%ld\n", error ));
  302. }
  303. }
  304. //
  305. // Start XACTSRV, if requested.
  306. //
  307. // *** This must be done AFTER the server driver is started, but
  308. // BEFORE sticky shares are recreated, otherwise downlevel print
  309. // shares are lost.
  310. //
  311. if ( SsData.ServerInfo599.sv599_acceptdownlevelapis ) {
  312. error = XsStartXactsrv( );
  313. if ( error != NO_ERROR ) {
  314. return error;
  315. }
  316. }
  317. //
  318. // Re-register the server's domain so we pick up the DNS name
  319. //
  320. SsSetDomainName();
  321. //
  322. // Create the default shares needed by the server.
  323. //
  324. error = CreateDefaultShares( );
  325. if ( error != NO_ERROR ) {
  326. return error;
  327. }
  328. IF_DEBUG(INITIALIZATION) {
  329. SS_PRINT(( "SsInitialize: default shares created.\n" ));
  330. }
  331. //
  332. // Complete loading the configuration--sticky shares and transports.
  333. // These must be done after the server FSP has started.
  334. //
  335. error = SsRecreateStickyShares( );
  336. if ( error != NO_ERROR ) {
  337. return error;
  338. }
  339. IF_DEBUG(INITIALIZATION) {
  340. SS_PRINT(( "SsInitialize: sticky shares reloaded.\n" ));
  341. }
  342. //
  343. // Set information used in server announcements.
  344. //
  345. SsSetExportedServerType( NULL, FALSE, FALSE );
  346. //
  347. // Server initialization was successful.
  348. //
  349. return NO_ERROR;
  350. } // SsInitialize
  351. NET_API_STATUS
  352. SsTerminate (
  353. VOID
  354. )
  355. /*++
  356. Routine Description:
  357. This routine sends the FSCTL_SRV_SHUTDOWN control code to the server
  358. FSD to tell it to terminate its FSP.
  359. Arguments:
  360. None.
  361. Return Value:
  362. None.
  363. --*/
  364. {
  365. NET_API_STATUS error;
  366. PNAME_LIST_ENTRY Service;
  367. PTRANSPORT_LIST_ENTRY Transport;
  368. //
  369. // Shut the server FSD/FSP down.
  370. //
  371. error = TerminateServer( );
  372. //
  373. // Shut down XACTSRV.
  374. //
  375. XsStopXactsrv( );
  376. //
  377. // Stop waiting for domain name changes
  378. //
  379. if( SsData.SsDomainNameChangeEvent ) {
  380. NetUnregisterDomainNameChangeNotification( SsData.SsDomainNameChangeEvent );
  381. SsData.SsDomainNameChangeEvent = NULL;
  382. }
  383. //
  384. // Delete security objects.
  385. //
  386. SsDeleteSecurityObjects( );
  387. //
  388. // Close the network announcement event.
  389. //
  390. if (SsData.SsAnnouncementEvent != NULL) {
  391. CloseHandle( SsData.SsAnnouncementEvent );
  392. SsData.SsAnnouncementEvent = NULL;
  393. }
  394. //
  395. // Close the local announcement event.
  396. //
  397. if (SsData.SsStatusChangedEvent != NULL) {
  398. CloseHandle( SsData.SsStatusChangedEvent );
  399. SsData.SsStatusChangedEvent = NULL;
  400. }
  401. //
  402. // Close the termination event.
  403. //
  404. if ( SsData.SsTerminationEvent != NULL ) {
  405. CloseHandle( SsData.SsTerminationEvent );
  406. SsData.SsTerminationEvent = NULL;
  407. }
  408. //
  409. // Free up the transport service list.
  410. //
  411. while( SsData.SsServerNameList != NULL ) {
  412. PNAME_LIST_ENTRY Service = SsData.SsServerNameList;
  413. while( Service->Transports != NULL ) {
  414. PTRANSPORT_LIST_ENTRY Next = Service->Transports->Next;
  415. MIDL_user_free( Service->Transports );
  416. Service->Transports = Next;
  417. }
  418. SsData.SsServerNameList = Service->Next;
  419. MIDL_user_free( Service );
  420. }
  421. //
  422. // Delete the server info resource.
  423. //
  424. if ( SsData.SsServerInfoResourceInitialized ) {
  425. RtlDeleteResource( &SsData.SsServerInfoResource );
  426. SsData.SsServerInfoResourceInitialized = FALSE;
  427. // Free any orphaned delete context's (caused by penetration style attacks)
  428. while( SrvShareDelContextHead != NULL )
  429. {
  430. PSHARE_DEL_CONTEXT pDelete = SrvShareDelContextHead;
  431. SrvShareDelContextHead = pDelete->Next;
  432. MIDL_user_free( pDelete );
  433. }
  434. DeleteCriticalSection( &ShareDelContextMutex );
  435. }
  436. //
  437. // Free up any string relate memory
  438. //
  439. FreeStrings() ;
  440. return error;
  441. } // SsTerminate
  442. NET_API_STATUS
  443. CreateDefaultShares (
  444. VOID
  445. )
  446. /*++
  447. Routine Description:
  448. This routine sends the NetShareAdd API to the server to add the
  449. default server shares using the data above.
  450. Arguments:
  451. None.
  452. Return Value:
  453. NET_API_STATUS - results of operation.
  454. --*/
  455. {
  456. NET_API_STATUS error;
  457. SHARE_INFO_2 shareInfo;
  458. SHARE_INFO shInfo;
  459. WCHAR diskShareName[3];
  460. WCHAR diskSharePath[4];
  461. ULONG i;
  462. DWORD diskMask;
  463. DWORD diskconfiguration;
  464. //
  465. // Create IPC$.
  466. //
  467. // !!! Need to verify the remarks for these default shares.
  468. //
  469. shareInfo.shi2_netname = IPC_SHARE_NAME;
  470. shareInfo.shi2_type = STYPE_IPC;
  471. shareInfo.shi2_remark = NULL;
  472. shareInfo.shi2_permissions = 0;
  473. shareInfo.shi2_max_uses = SHI_USES_UNLIMITED;
  474. shareInfo.shi2_current_uses = 0;
  475. shareInfo.shi2_path = NULL;
  476. shareInfo.shi2_passwd = NULL;
  477. shInfo.ShareInfo2 = &shareInfo;
  478. error = NetrShareAdd( NULL, 2, &shInfo, NULL );
  479. if ( error != NO_ERROR ) {
  480. IF_DEBUG(INITIALIZATION_ERRORS) {
  481. SS_PRINT(( "CreateDefaultShares: failed to add " FORMAT_LPWSTR
  482. ": %X\n", shareInfo.shi2_netname, error ));
  483. }
  484. } else {
  485. IF_DEBUG(INITIALIZATION) {
  486. SS_PRINT(( "CreateDefaultShares: added default share "
  487. FORMAT_LPWSTR "\n", shareInfo.shi2_netname, error ));
  488. }
  489. }
  490. //
  491. // If this is a workstation, and the AutoShareWks key is set to TRUE then
  492. // automatically create the admin$ and drive$ shares.
  493. //
  494. //
  495. // If this is a server, and the AutoShareServer key is set to TRUE then
  496. // automatically create the admin$ and drive$ shares.
  497. //
  498. if( (SsData.ServerInfo598.sv598_producttype == NtProductWinNt &&
  499. SsData.ServerInfo598.sv598_autosharewks) ||
  500. (SsData.ServerInfo598.sv598_producttype != NtProductWinNt &&
  501. SsData.ServerInfo598.sv598_autoshareserver ) ) {
  502. //
  503. // Create ADMIN$.
  504. //
  505. shareInfo.shi2_netname = ADMIN_SHARE_NAME;
  506. shareInfo.shi2_type = STYPE_DISKTREE;
  507. shareInfo.shi2_remark = NULL;
  508. shareInfo.shi2_permissions = 1;
  509. shareInfo.shi2_max_uses = SHI_USES_UNLIMITED;
  510. shareInfo.shi2_current_uses = 0;
  511. shareInfo.shi2_path = NULL;
  512. shareInfo.shi2_passwd = NULL;
  513. error = NetrShareAdd( NULL, 2, &shInfo, NULL );
  514. if ( error != NO_ERROR ) {
  515. IF_DEBUG(INITIALIZATION_ERRORS) {
  516. SS_PRINT(( "CreateDefaultShares: failed to add " FORMAT_LPWSTR
  517. ": %X\n", shareInfo.shi2_netname, error ));
  518. }
  519. } else {
  520. IF_DEBUG(INITIALIZATION) {
  521. SS_PRINT(( "CreateDefaultShares: added default share "
  522. FORMAT_LPWSTR "\n", shareInfo.shi2_netname, error ));
  523. }
  524. }
  525. //
  526. // Loop through available drives, creating an admin share for each
  527. // one. Note that we allow "holes" in the drive letter space.
  528. //
  529. diskShareName[0] = 'A';
  530. diskShareName[1] = '$';
  531. diskShareName[2] = '\0';
  532. diskSharePath[0] = diskShareName[0];
  533. diskSharePath[1] = ':';
  534. diskSharePath[2] = '\\';
  535. diskSharePath[3] = '\0';
  536. shareInfo.shi2_netname = diskShareName;
  537. shareInfo.shi2_type = STYPE_DISKTREE;
  538. shareInfo.shi2_remark = SsDiskAdminShareRemark;
  539. shareInfo.shi2_permissions = 1;
  540. shareInfo.shi2_max_uses = SHI_USES_UNLIMITED;
  541. shareInfo.shi2_current_uses = 0;
  542. shareInfo.shi2_path = diskSharePath;
  543. shareInfo.shi2_passwd = NULL;
  544. diskconfiguration = DiscoverDrives();
  545. for ( i = 0, diskMask = 0x80000000;
  546. (i < SRVSVC_MAX_NUMBER_OF_DISKS) && (diskShareName[0] <= 'Z');
  547. i++, diskShareName[0]++, diskSharePath[0]++, diskMask >>= 1 ) {
  548. if ( (diskconfiguration & diskMask) != 0) {
  549. error = NetrShareAdd( NULL, 2, &shInfo, NULL );
  550. if ( error != NO_ERROR ) {
  551. IF_DEBUG(INITIALIZATION_ERRORS) {
  552. SS_PRINT(( "CreateDefaultShares: failed to add "
  553. FORMAT_LPWSTR ": %X\n",
  554. shareInfo.shi2_netname, error ));
  555. }
  556. } else {
  557. IF_DEBUG(INITIALIZATION) {
  558. SS_PRINT(( "CreateDefaultShares: added default share "
  559. FORMAT_LPWSTR "\n",
  560. shareInfo.shi2_netname, error ));
  561. }
  562. }
  563. }
  564. }
  565. }
  566. return NO_ERROR;
  567. } // CreateDefaultShares
  568. DWORD
  569. DiscoverDrives (
  570. VOID
  571. )
  572. /*++
  573. Routine Description:
  574. This routine returns a bit mask representing the local drives present
  575. on the system.
  576. Arguments:
  577. None.
  578. Return Value:
  579. DrivesAvailable - A 32 bit field representing the available drives on
  580. the system. The MSB represents drive A, the next represents drive
  581. B, etc. The extra 6 bits are currently unsed.
  582. --*/
  583. {
  584. WCHAR rootDirPath[4];
  585. WCHAR driveLetter;
  586. DWORD drivesAvailable = 0;
  587. DWORD driveMask = 0x80000000;
  588. UINT driveType;
  589. rootDirPath[1] = ':';
  590. rootDirPath[2] = '\\';
  591. rootDirPath[3] = '\0';
  592. for ( driveLetter = 'A';
  593. driveLetter <= 'Z';
  594. driveLetter++ ) {
  595. rootDirPath[0] = driveLetter;
  596. driveType = SsGetDriveType( rootDirPath );
  597. //
  598. // We only put fixed disk drives into the mask. We used to put
  599. // removables, CD-ROMs, and RAM disks into the list. But this
  600. // list is used for two purposes: creation of X$ shares (for
  601. // backup purposes), and free disk space checking (for admin
  602. // purposes). Neither of these uses applies very well to these
  603. // devices.
  604. //
  605. if ( driveType == DRIVE_FIXED
  606. //|| driveType == DRIVE_REMOVABLE
  607. //|| driveType == DRIVE_CDROM
  608. //|| driveType == DRIVE_RAMDISK
  609. ) {
  610. //
  611. // This is a valid drive letter
  612. //
  613. drivesAvailable |= driveMask;
  614. }
  615. //
  616. // Update drive mask for the next drive
  617. //
  618. driveMask /= 2;
  619. }
  620. return drivesAvailable;
  621. }
  622. VOID
  623. InitializeDefaultData(
  624. VOID
  625. )
  626. /*++
  627. Routine Description:
  628. This routine sets up the default data in the server service by using
  629. the values in srvconfg.h.
  630. Arguments:
  631. None.
  632. Return Value:
  633. None.
  634. --*/
  635. {
  636. NET_API_STATUS error;
  637. USHORT i;
  638. OSVERSIONINFOEX VersionInformation;
  639. NT_PRODUCT_TYPE ProductType;
  640. WCHAR szNumber[ sizeof( SsData.szVersionNumber ) / sizeof( WCHAR ) ], *p;
  641. //
  642. // Loop through all the defined fields, setting them as we go.
  643. //
  644. for ( i = 0; SsServerInfoFields[i].FieldName != NULL; i++ ) {
  645. error = SsSetField(
  646. &SsServerInfoFields[i],
  647. &SsServerInfoFields[i].DefaultValue,
  648. FALSE,
  649. NULL
  650. );
  651. SS_ASSERT( error == NO_ERROR );
  652. }
  653. SsData.NumberOfPrintShares = 0;
  654. //
  655. // Get the system version and product name
  656. //
  657. VersionInformation.dwOSVersionInfoSize = sizeof( VersionInformation );
  658. i = (USHORT)GetVersionEx( (LPOSVERSIONINFO)&VersionInformation );
  659. SS_ASSERT( i == TRUE );
  660. SsData.ServerInfo102.sv102_version_major = VersionInformation.dwMajorVersion;
  661. SsData.ServerInfo102.sv102_version_minor = VersionInformation.dwMinorVersion;
  662. wcscpy( SsData.ServerProductName, SERVER_PRODUCT_NAME );
  663. //
  664. // Convert the version number to a version number string...
  665. //
  666. szNumber[ sizeof( szNumber ) / sizeof( szNumber[0] ) - 1 ] = L'\0';
  667. for( p = &szNumber[ sizeof( szNumber ) / sizeof( szNumber[0] ) - 2 ]; p > &szNumber[0]; p-- ) {
  668. *p = L"0123456789"[ VersionInformation.dwMinorVersion % 10 ];
  669. VersionInformation.dwMinorVersion /= 10;
  670. if( VersionInformation.dwMinorVersion == 0 )
  671. break;
  672. }
  673. *(--p) = L'.';
  674. do {
  675. *(--p) = L"0123456789"[ VersionInformation.dwMajorVersion % 10 ];
  676. VersionInformation.dwMajorVersion /= 10;
  677. } while( VersionInformation.dwMajorVersion && p > &szNumber[0] );
  678. if( VersionInformation.wSuiteMask & VER_SUITE_PERSONAL )
  679. {
  680. // Turn off auto-shares by default on Personal
  681. SsData.ServerInfo598.sv598_autoshareserver = FALSE;
  682. SsData.ServerInfo598.sv598_autosharewks = FALSE;
  683. }
  684. //
  685. // ... and store it in SsData
  686. //
  687. wcscpy( SsData.szVersionNumber, p );
  688. //
  689. // Change certain defaults for Workstations
  690. if( RtlGetNtProductType( &ProductType ) )
  691. {
  692. if( ProductType == NtProductWinNt )
  693. {
  694. SsData.ServerInfo599.sv599_diskspacethreshold = 0;
  695. SsData.ServerInfo598.sv598_disabledos = TRUE;
  696. }
  697. }
  698. // Change defaults for Embedded
  699. if( IsEmbedded() )
  700. {
  701. MINIMIZE( SsData.ServerInfo102.sv102_users, MAX_USERS_EMBEDDED );
  702. }
  703. } // InitializeDefaultData
  704. NET_API_STATUS
  705. InitializeServer (
  706. VOID
  707. )
  708. /*++
  709. Routine Description:
  710. This routine sends the FSCTL_SRV_STARTUP control code to the server
  711. FSD to tell it to start and initialize its FSP.
  712. Arguments:
  713. None.
  714. Return Value:
  715. NET_API_STATUS - results of operation.
  716. --*/
  717. {
  718. NET_API_STATUS error;
  719. PSERVER_REQUEST_PACKET srp;
  720. SS_ASSERT( !SsData.SsServerFspStarted );
  721. //
  722. // Load the server driver.
  723. //
  724. error = LoadServer( );
  725. if ( error != NO_ERROR ) {
  726. return error;
  727. }
  728. //
  729. // Get an SRP and set it up with the appropriate level.
  730. //
  731. srp = SsAllocateSrp( );
  732. if ( srp == NULL ) {
  733. UnloadServer();
  734. return ERROR_NOT_ENOUGH_MEMORY;
  735. }
  736. srp->Level = (ULONG)SS_STARTUP_LEVEL;
  737. //
  738. // Pass domain name to the server.
  739. //
  740. RtlInitUnicodeString( &srp->Name1, SsData.DomainNameBuffer );
  741. //
  742. // Pass server name to the server.
  743. //
  744. RtlInitUnicodeString( &srp->Name2, SsData.ServerNameBuffer );
  745. //
  746. // Send the request on to the server.
  747. //
  748. error = SsServerFsControl(
  749. FSCTL_SRV_STARTUP,
  750. srp,
  751. &SsData.ServerInfo102,
  752. sizeof(SERVER_INFO_102) + sizeof(SERVER_INFO_599) +
  753. sizeof(SERVER_INFO_598)
  754. );
  755. if ( error == NO_ERROR ) {
  756. SsData.SsServerFspStarted = TRUE;
  757. } else {
  758. UnloadServer();
  759. }
  760. //
  761. // Free the SRP and return.
  762. //
  763. SsFreeSrp( srp );
  764. return error;
  765. } // InitializeServer
  766. NET_API_STATUS
  767. StartPnpNotifications (
  768. VOID
  769. )
  770. /*++
  771. Routine Description:
  772. This routine sends the FSCTL_SRV_BEGIN_PNP_NOTIFICATIONS control code to the server
  773. FSD to tell it to start monitoring transport PNP notifications
  774. Arguments:
  775. None.
  776. Return Value:
  777. NET_API_STATUS - results of operation.
  778. --*/
  779. {
  780. NET_API_STATUS error;
  781. //
  782. // Send the request on to the server.
  783. //
  784. error = SsServerFsControl(
  785. FSCTL_SRV_BEGIN_PNP_NOTIFICATIONS,
  786. NULL,
  787. NULL,
  788. 0
  789. );
  790. IF_DEBUG(INITIALIZATION) {
  791. if( error != NO_ERROR ) {
  792. SS_PRINT(( "StartPnpNotifications: error %X\n", error ));
  793. }
  794. }
  795. return error;
  796. } // StartPnpNotifications
  797. NET_API_STATUS
  798. LoadServer (
  799. VOID
  800. )
  801. {
  802. NTSTATUS status;
  803. NET_API_STATUS error;
  804. LPWSTR registryPathBuffer;
  805. UNICODE_STRING registryPath;
  806. ULONG privileges[1];
  807. LPWSTR subString[1];
  808. IF_DEBUG(INITIALIZATION) {
  809. SS_PRINT(( "LoadServer: entered\n" ));
  810. }
  811. registryPathBuffer = (LPWSTR)MIDL_user_allocate(
  812. sizeof(SERVICE_REGISTRY_KEY) +
  813. sizeof(SERVER_DRIVER_NAME) +
  814. sizeof(WCHAR) // for null
  815. );
  816. if ( registryPathBuffer == NULL ) {
  817. IF_DEBUG(INITIALIZATION_ERRORS) {
  818. SS_PRINT(( "LoadServer: Unable to allocate memory\n" ));
  819. }
  820. return ERROR_NOT_ENOUGH_MEMORY;
  821. }
  822. privileges[0] = SE_LOAD_DRIVER_PRIVILEGE;
  823. error = NetpGetPrivilege( 1, privileges );
  824. if ( error != NO_ERROR ) {
  825. IF_DEBUG(INITIALIZATION_ERRORS) {
  826. SS_PRINT(( "LoadServer: Unable to enable privilege: %ld\n",
  827. error ));
  828. }
  829. MIDL_user_free( registryPathBuffer );
  830. return error;
  831. }
  832. wcscpy( registryPathBuffer, SERVICE_REGISTRY_KEY );
  833. wcscat( registryPathBuffer, SERVER_DRIVER_NAME );
  834. RtlInitUnicodeString( &registryPath, registryPathBuffer );
  835. status = NtLoadDriver( &registryPath );
  836. MIDL_user_free( registryPathBuffer );
  837. if ( status == STATUS_IMAGE_ALREADY_LOADED ) {
  838. status = STATUS_SUCCESS;
  839. }
  840. if ( !NT_SUCCESS(status) ) {
  841. IF_DEBUG(INITIALIZATION_ERRORS) {
  842. SS_PRINT(( "LoadServer: Unable to load driver: %lx\n",
  843. status ));
  844. }
  845. subString[0] = SERVER_DRIVER_NAME;
  846. SsLogEvent(
  847. EVENT_SRV_CANT_LOAD_DRIVER,
  848. 1,
  849. subString,
  850. status
  851. );
  852. error = RtlNtStatusToDosError(status);
  853. } else {
  854. //
  855. // Get a handle to the server.
  856. //
  857. error = SsOpenServer();
  858. if ( error != NO_ERROR ) {
  859. UnloadServer();
  860. }
  861. }
  862. IF_DEBUG(INITIALIZATION) {
  863. SS_PRINT(( "LoadServer: returning\n" ));
  864. }
  865. NetpReleasePrivilege( );
  866. return error;
  867. } // LoadServer
  868. NET_API_STATUS
  869. ConvertStringToTransportAddress (
  870. IN PUNICODE_STRING InputName,
  871. OUT CHAR TransportAddress[ MAX_PATH ],
  872. OUT PULONG TransportAddressLength
  873. )
  874. {
  875. OEM_STRING computerName;
  876. if( InputName == NULL || InputName->Length == 0 ) {
  877. RtlCopyMemory( TransportAddress,
  878. SsData.SsServerTransportAddress,
  879. SsData.SsServerTransportAddressLength );
  880. *TransportAddressLength = SsData.SsServerTransportAddressLength;
  881. return NO_ERROR;
  882. }
  883. if( InputName->Length > (MAX_PATH - 1 ) * sizeof( WCHAR ) ) {
  884. return ERROR_INVALID_PARAMETER;
  885. }
  886. //
  887. // Write directly to the output buffer
  888. //
  889. computerName.Buffer = TransportAddress;
  890. computerName.MaximumLength = MAX_PATH;
  891. //
  892. // Convert To Oem Name
  893. //
  894. (VOID) RtlUpcaseUnicodeStringToOemString(
  895. &computerName,
  896. InputName,
  897. FALSE
  898. );
  899. //
  900. // Make sure it is exactly NETBIOS_NAME_LEN characters
  901. //
  902. if( computerName.Length < NETBIOS_NAME_LEN ) {
  903. RtlCopyMemory( TransportAddress + computerName.Length,
  904. " ",
  905. NETBIOS_NAME_LEN - computerName.Length
  906. );
  907. *TransportAddressLength = NETBIOS_NAME_LEN;
  908. } else {
  909. *TransportAddressLength = NETBIOS_NAME_LEN;
  910. }
  911. return NO_ERROR;
  912. } // ConvertStringToTransportAddress
  913. VOID
  914. SsSetDomainName (
  915. VOID
  916. )
  917. /*++
  918. Routine Description:
  919. Calls NetpGetDomainName to determine the domain name the server
  920. should use. Ensure this domain name is reflected throughout the server service and
  921. srv.sys
  922. Arguments:
  923. None.
  924. Return Value:
  925. None.
  926. --*/
  927. {
  928. NET_API_STATUS error;
  929. LPWSTR domainName = NULL;
  930. LPWSTR dnsDomainName = NULL;
  931. LPWSTR fullDnsDomainName = NULL;
  932. PNAME_LIST_ENTRY service;
  933. BOOLEAN IsWorkgroup;
  934. DWORD dwError;
  935. dwError = NetpGetDomainNameExEx( &domainName, &fullDnsDomainName, &IsWorkgroup );
  936. if( dwError != NO_ERROR )
  937. {
  938. LPWSTR subString[2];
  939. subString[0] = SsData.DomainNameBuffer;
  940. subString[1] = L"????";
  941. SsLogEvent(
  942. EVENT_SRV_CANT_CHANGE_DOMAIN_NAME,
  943. 2,
  944. subString,
  945. dwError
  946. );
  947. return;
  948. }
  949. if( fullDnsDomainName )
  950. {
  951. // Strip the name domain.foo.com down do just "domain"
  952. dnsDomainName = wcstok( fullDnsDomainName, L"." );
  953. }
  954. RtlAcquireResourceExclusive( &SsData.SsServerInfoResource, TRUE );
  955. if( SsData.SsServerFspStarted && SsData.DomainNameBuffer[0] ) {
  956. SERVER_REQUEST_PACKET srp;
  957. NTSTATUS status;
  958. //
  959. // If we used to have a domain name, change the service list to the
  960. // new domain name
  961. //
  962. for( service = SsData.SsServerNameList; service != NULL; service = service->Next ) {
  963. if( !STRCMPI( service->DomainName, SsData.DomainNameBuffer ) ) {
  964. STRCPY( service->DomainName, domainName );
  965. }
  966. }
  967. srp.Name1.Length = wcslen( SsData.DomainNameBuffer ) * sizeof( WCHAR );
  968. srp.Name1.MaximumLength = srp.Name1.Length;
  969. srp.Name1.Buffer = SsData.DomainNameBuffer;
  970. srp.Name2.Length = wcslen( domainName ) * sizeof( WCHAR );
  971. srp.Name2.MaximumLength = srp.Name2.Length;
  972. srp.Name2.Buffer = domainName;
  973. //
  974. // Tell the SMB server about this change
  975. //
  976. status = SsServerFsControl( FSCTL_SRV_CHANGE_DOMAIN_NAME, &srp, NULL, 0 );
  977. //
  978. // If we are unable to change the domain name, log an error
  979. //
  980. if( !NT_SUCCESS( status ) ) {
  981. LPWSTR subString[2];
  982. subString[0] = SsData.DomainNameBuffer;
  983. subString[1] = domainName;
  984. SsLogEvent(
  985. EVENT_SRV_CANT_CHANGE_DOMAIN_NAME,
  986. 2,
  987. subString,
  988. status
  989. );
  990. }
  991. if( fullDnsDomainName && !IsWorkgroup )
  992. {
  993. //
  994. // Tell the SMB server about the DNS Domain Name change too
  995. //
  996. srp.Name2.Length = wcslen( dnsDomainName ) * sizeof( WCHAR );
  997. srp.Name2.MaximumLength = srp.Name2.Length;
  998. srp.Name2.Buffer = dnsDomainName;
  999. srp.Name1.Length = wcslen( domainName ) * sizeof( WCHAR );
  1000. srp.Name1.MaximumLength = srp.Name2.Length;
  1001. srp.Name1.Buffer = domainName;
  1002. //
  1003. // Tell the SMB server about this change
  1004. //
  1005. status = SsServerFsControl( FSCTL_SRV_CHANGE_DNS_DOMAIN_NAME, &srp, NULL, 0 );
  1006. //
  1007. // If we are unable to change the domain name, log an error
  1008. //
  1009. if( !NT_SUCCESS( status ) ) {
  1010. LPWSTR subString[2];
  1011. subString[0] = SsData.DomainNameBuffer;
  1012. subString[1] = domainName;
  1013. SsLogEvent(
  1014. EVENT_SRV_CANT_CHANGE_DOMAIN_NAME,
  1015. 2,
  1016. subString,
  1017. status
  1018. );
  1019. }
  1020. }
  1021. }
  1022. //
  1023. // Copy the name into our name buffer.
  1024. //
  1025. STRNCPY( SsData.DomainNameBuffer, domainName, MAX_PATH);
  1026. RtlReleaseResource( &SsData.SsServerInfoResource );
  1027. //
  1028. // Free the storage allocated by NetpGetComputerName.
  1029. //
  1030. (VOID)NetApiBufferFree( domainName );
  1031. if( fullDnsDomainName )
  1032. {
  1033. (VOID)NetApiBufferFree( fullDnsDomainName );
  1034. }
  1035. IF_DEBUG(INITIALIZATION) {
  1036. SS_PRINT(( "SsSetDomainName: domain name set to " FORMAT_LPWSTR
  1037. "(could be overridden later!)\n",
  1038. SsData.DomainNameBuffer ));
  1039. }
  1040. } // SsSetDomainName
  1041. VOID
  1042. SetServerName (
  1043. VOID
  1044. )
  1045. /*++
  1046. Routine Description:
  1047. Calls NetpGetComputerName to determine the name the server should use
  1048. to register itself on the network.
  1049. Arguments:
  1050. None.
  1051. Return Value:
  1052. None.
  1053. --*/
  1054. {
  1055. NET_API_STATUS error;
  1056. LPWSTR computerName;
  1057. //
  1058. // Get the computer name.
  1059. //
  1060. error = NetpGetComputerName( &computerName );
  1061. SS_ASSERT( error == NO_ERROR );
  1062. //
  1063. // Copy the name into our name buffer. This name is returned to
  1064. // our apis.
  1065. //
  1066. STRCPY( SsData.ServerNameBuffer, computerName );
  1067. //
  1068. // Free the storage allocated by NetpGetComputerName.
  1069. //
  1070. (void) NetApiBufferFree( computerName );
  1071. //
  1072. // Uppercase the server name. This name is used for announcements.
  1073. //
  1074. {
  1075. UNICODE_STRING serverName;
  1076. SsData.ServerAnnounceName.Length =
  1077. serverName.Length =
  1078. (USHORT) (STRLEN( SsData.ServerNameBuffer ) * sizeof(WCHAR));
  1079. SsData.ServerAnnounceName.MaximumLength =
  1080. serverName.MaximumLength =
  1081. (USHORT) (serverName.Length + sizeof(WCHAR));
  1082. serverName.Buffer = SsData.ServerNameBuffer;
  1083. SsData.ServerAnnounceName.Buffer = SsData.AnnounceNameBuffer;
  1084. (VOID)RtlUpcaseUnicodeString(
  1085. &SsData.ServerAnnounceName,
  1086. &serverName,
  1087. FALSE
  1088. );
  1089. //
  1090. // Make the server name in Netbios format.
  1091. //
  1092. error = ConvertStringToTransportAddress(
  1093. &serverName,
  1094. SsData.SsServerTransportAddress,
  1095. &SsData.SsServerTransportAddressLength
  1096. );
  1097. SS_ASSERT( error == NO_ERROR );
  1098. }
  1099. IF_DEBUG(INITIALIZATION) {
  1100. SS_PRINT(( "SetServerName: server name set to " FORMAT_LPWSTR
  1101. " (could be overridden later!)\n",
  1102. SsData.ServerNameBuffer ));
  1103. }
  1104. return;
  1105. } // SetServerName
  1106. NET_API_STATUS
  1107. TerminateServer (
  1108. VOID
  1109. )
  1110. /*++
  1111. Routine Description:
  1112. This routine sends the FSCTL_SRV_SHUTDOWN control code to the server
  1113. FSD to tell it to shutdown operations.
  1114. Arguments:
  1115. None.
  1116. Return Value:
  1117. None.
  1118. --*/
  1119. {
  1120. NET_API_STATUS error = NO_ERROR;
  1121. if ( SsData.SsServerFspStarted ) {
  1122. SsData.SsServerFspStarted = FALSE;
  1123. //
  1124. // Send the request on to the server.
  1125. //
  1126. error = SsServerFsControl(
  1127. FSCTL_SRV_SHUTDOWN,
  1128. NULL,
  1129. NULL,
  1130. 0
  1131. );
  1132. if ( (error != NO_ERROR) &&
  1133. (error != ERROR_SERVER_HAS_OPEN_HANDLES) ) {
  1134. IF_DEBUG(TERMINATION_ERRORS) {
  1135. SS_PRINT(( "TerminateServer: FSCTL_SRV_SHUTDOWN failed: %ld\n",
  1136. error ));
  1137. }
  1138. }
  1139. //
  1140. // Unload the server driver, unless there are other open handles
  1141. // to the server. We don't unload the driver in this case
  1142. // because the driver won't actually go away until the
  1143. // additional handles are closed, so the driver will not be
  1144. // fully unloaded. This would cause a subsequent server startup
  1145. // to fail.
  1146. //
  1147. if ( error != ERROR_SERVER_HAS_OPEN_HANDLES ) {
  1148. IF_DEBUG(TERMINATION) {
  1149. SS_PRINT(( "TerminateServer: Unloading server\n" ));
  1150. }
  1151. UnloadServer( );
  1152. }
  1153. }
  1154. //
  1155. // Close the handle to the server.
  1156. //
  1157. SsCloseServer( );
  1158. return error;
  1159. } // TerminateServer
  1160. VOID
  1161. UnloadServer (
  1162. VOID
  1163. )
  1164. {
  1165. NTSTATUS status;
  1166. NET_API_STATUS error;
  1167. LPWSTR registryPathBuffer;
  1168. UNICODE_STRING registryPath;
  1169. ULONG privileges[1];
  1170. LPWSTR subString[1];
  1171. registryPathBuffer = (LPWSTR)MIDL_user_allocate(
  1172. sizeof(SERVICE_REGISTRY_KEY) +
  1173. sizeof(SERVER_DRIVER_NAME) +
  1174. sizeof(WCHAR) // for null
  1175. );
  1176. if ( registryPathBuffer == NULL ) {
  1177. IF_DEBUG(TERMINATION_ERRORS) {
  1178. SS_PRINT(( "UnloadServer: Unable to allocate memory\n" ));
  1179. }
  1180. return;
  1181. }
  1182. privileges[0] = SE_LOAD_DRIVER_PRIVILEGE;
  1183. error = NetpGetPrivilege( 1, privileges );
  1184. if ( error != NO_ERROR ) {
  1185. IF_DEBUG(TERMINATION_ERRORS) {
  1186. SS_PRINT(( "UnloadServer: Unable to enable privilege: %ld\n",
  1187. error ));
  1188. }
  1189. MIDL_user_free( registryPathBuffer );
  1190. return;
  1191. }
  1192. wcscpy( registryPathBuffer, SERVICE_REGISTRY_KEY );
  1193. wcscat( registryPathBuffer, SERVER_DRIVER_NAME );
  1194. RtlInitUnicodeString( &registryPath, registryPathBuffer );
  1195. status = NtUnloadDriver( &registryPath );
  1196. MIDL_user_free( registryPathBuffer );
  1197. NetpReleasePrivilege( );
  1198. if ( !NT_SUCCESS(status) ) {
  1199. IF_DEBUG(TERMINATION_ERRORS) {
  1200. SS_PRINT(( "UnloadServer: Unable to unload driver: %lx\n",
  1201. status ));
  1202. }
  1203. subString[0] = SERVER_DRIVER_NAME;
  1204. SsLogEvent(
  1205. EVENT_SRV_CANT_UNLOAD_DRIVER,
  1206. 1,
  1207. subString,
  1208. status
  1209. );
  1210. }
  1211. return;
  1212. } // UnloadServer
  1213. VOID
  1214. InitializeStrings(
  1215. VOID
  1216. )
  1217. /*++
  1218. Routine Description:
  1219. Retrieve internationalizable strings from NETMSG.DLL. They
  1220. are used for share comments for IPC$, ADMIN$, C$, etc.
  1221. Routine does not report any errors. If there are problems,
  1222. the strings will be empty ones.
  1223. FreeStrings should be called to free the memory allocated
  1224. by format message and the
  1225. Arguments:
  1226. None.
  1227. Return Value:
  1228. None.
  1229. --*/
  1230. {
  1231. DWORD dwRet, dwFlags ;
  1232. HMODULE hModule ;
  1233. //
  1234. // init the strings to the default empty remark.
  1235. //
  1236. SsAdminShareRemark = SsDefaultRemark ;
  1237. SsIPCShareRemark = SsDefaultRemark ;
  1238. SsDiskAdminShareRemark = SsDefaultRemark ;
  1239. //
  1240. // load NETMSG.DLL - if we cannot, just return.
  1241. //
  1242. hModule = LoadLibrary(NETMSG_DLL) ;
  1243. if(!hModule)
  1244. return ;
  1245. //
  1246. // hit FormatMessage 3 times for the real thing...
  1247. //
  1248. dwFlags = FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_HMODULE ;
  1249. dwRet = FormatMessage(dwFlags,
  1250. hModule,
  1251. APE2_SERVER_IPC_SHARE_REMARK,
  1252. 0,
  1253. (LPWSTR) &SsIPCShareRemark,
  1254. 1,
  1255. NULL) ;
  1256. if (dwRet == 0)
  1257. SsIPCShareRemark = SsDefaultRemark ;
  1258. dwRet = FormatMessage(dwFlags,
  1259. hModule,
  1260. APE2_SERVER_ADMIN_SHARE_REMARK,
  1261. 0,
  1262. (LPWSTR) &SsAdminShareRemark,
  1263. 1,
  1264. NULL) ;
  1265. if (dwRet == 0)
  1266. SsAdminShareRemark = SsDefaultRemark ;
  1267. dwRet = FormatMessage(dwFlags,
  1268. hModule,
  1269. APE2_SERVER_DISK_ADMIN_SHARE_REMARK,
  1270. 0,
  1271. (LPWSTR) &SsDiskAdminShareRemark,
  1272. 1,
  1273. NULL) ;
  1274. if (dwRet == 0)
  1275. SsDiskAdminShareRemark = SsDefaultRemark ;
  1276. FreeLibrary(hModule) ;
  1277. }
  1278. VOID
  1279. FreeStrings(
  1280. VOID
  1281. )
  1282. /*++
  1283. Routine Description:
  1284. Free the memory used by server comment strings (allocated by
  1285. FormatMessage).
  1286. Arguments:
  1287. None.
  1288. Return Value:
  1289. None.
  1290. --*/
  1291. {
  1292. //
  1293. // as long as the strings do not point to the default (static data),
  1294. // free them.
  1295. //
  1296. if (SsAdminShareRemark && SsAdminShareRemark != SsDefaultRemark)
  1297. LocalFree(SsAdminShareRemark) ;
  1298. SsAdminShareRemark = SsDefaultRemark ;
  1299. if (SsIPCShareRemark && SsIPCShareRemark != SsDefaultRemark)
  1300. LocalFree(SsIPCShareRemark) ;
  1301. SsIPCShareRemark = SsDefaultRemark ;
  1302. if (SsDiskAdminShareRemark && SsDiskAdminShareRemark != SsDefaultRemark)
  1303. LocalFree(SsDiskAdminShareRemark) ;
  1304. SsDiskAdminShareRemark = SsDefaultRemark ;
  1305. }