Leaked source code of windows server 2003
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.

2963 lines
93 KiB

  1. /*++
  2. Copyright (c) 1992 Microsoft Corporation
  3. Module Name:
  4. depend.cxx
  5. Abstract:
  6. This module contains routines which handle service start and
  7. stop dependencies:
  8. ScInitAutoStart
  9. ScAutoStartServices
  10. ScStartServiceAndDependencies
  11. ScSetServiceStartRequest
  12. ScMarkGroupStartNow
  13. RI_ScGetCurrentGroupStateW
  14. ScStartMarkedServices
  15. ScHandleServiceFailure
  16. ScDependenciesStarted
  17. ScLookForHungServices
  18. ScHandleBadDependencies
  19. ScServiceToStartDependOn
  20. ScNotifyChangeState
  21. ScDependentsStopped
  22. ScEnumDependents
  23. ScFoundDuplicateDependent
  24. ScInHardwareProfile
  25. Author:
  26. Rita Wong (ritaw) 03-Apr-1992
  27. Environment:
  28. Win32
  29. Revision History:
  30. 11-Jun-2000 JSchwart
  31. Add no-reboot support for machine name change during OOBE
  32. setup and initial boot
  33. 08-Jan-1997 AnirudhS
  34. Remove I_ScGetCurrentGroupStateW as it is no longer used. In
  35. ScStartServiceAndDependencies, get an exclusive grouplist lock
  36. once more, instead of a shared one.
  37. 11-Jun-1996 AnirudhS
  38. During setup/upgrade, don't event-log failure of a service to start
  39. due to a dependent service not starting. (The most common case of
  40. this is a service that runs in a domain account, and hence has an
  41. implicit dependency on netlogon, which is disabled during setup.)
  42. 03-Nov-1995 AnirudhS
  43. Don't try to start a service that isn't in the current hardware
  44. profile.
  45. 30-Oct-1995 AnirudhS
  46. ScStartMarkedServices: If ScStartService says that a service is
  47. already running, treat this as a success.
  48. 15-Aug-1995 AnirudhS
  49. Added I_ScGetCurrentGroupStateW.
  50. Changed while loops to for loops and if stmts to switch stmts for
  51. improved readability.
  52. 16-Aug-1994 Danl
  53. ScLookForHungServices: If a long waitHint was passed in, the sleep
  54. time would be set to a huge number (like 4.9 days). This was
  55. changed so that if the waitHint is over 100 seconds, then the
  56. sleep time is limited to 10 seconds, but the number of iterations
  57. for the polling goes up.
  58. 09-Jun-1994 Danl
  59. Begin working on making sure NetLogon is started if we are
  60. going to start a service that runs in an account. This requires
  61. making a dependency on NetLogon.
  62. 21-Apr-1992 JohnRo
  63. Use SC_LOG0(), FORMAT_ equates, etc
  64. 03-Apr-1992 ritaw
  65. created
  66. --*/
  67. #include "precomp.hxx"
  68. #include <string.h> // memcpy
  69. #include <stdlib.h> // wcslen
  70. #include <winerror.h>
  71. #include <lmcons.h> // NET_API_STATUS
  72. #include <srvann.h> // I_ScGetCurrentGroupStateW and related definitions
  73. #include <cfgmgr32.h> // PNP manager functions
  74. #include <pnp.h> // PNP manager functions, server side
  75. #include <cfgmgrp.h> // PNP manager functions, server side, internal (PNP_GET_HWPROFFLAGS)
  76. #include <regstr.h> // CSCONFIGFLAG_ constants
  77. #include <tstr.h> // WCSSIZE
  78. #include <sclib.h> // ScCopyStringToBufferW
  79. #include <svcslib.h> // SetupInProgress
  80. #include <winsvcp.h> // OOBE setup event names
  81. #include "start.h" // ScStartService
  82. #include "depend.h"
  83. #include "info.h" // ScQueryServiceStatus
  84. #include "bootcfg.h" // ScRevertToLastKnownGood
  85. #include "driver.h" // ScGetDriverStatus
  86. #include "lockapi.h" // ScLockDatabase
  87. #include "account.h" // SC_LOCAL_SYSTEM_USER_NAME
  88. #include "scconfig.h" // ScRegCloseKey
  89. #define SERVICE_START_TIMEOUT 80000 // 80 seconds
  90. #define LENGTH(array) (sizeof(array)/sizeof((array)[0]))
  91. #define GROUP_NOT_STARTED 0x00000000
  92. #define GROUP_ONE_STARTED 0x00000001
  93. #define GROUP_START_FAIL 0x00000002
  94. #define REG_KEY_SETUP_ALLOW_START L"System\\Setup\\AllowStart"
  95. //
  96. // TDI GROUP SPECIAL: The PNP_TDI group is treated as a subgroup of
  97. // the TDI group for dependency purposes (though not for group start
  98. // ordering purposes). This is implemented via the following macros.
  99. // A service BELONGS_TO a group either if it is a member of that group,
  100. // or if the group is the TDI group and the service is a member of the
  101. // PNP_TDI group.
  102. // IS_SUBGROUP returns true if Group1 is equal to or a subgroup of Group2.
  103. //
  104. #define IS_SUBGROUP(Group1, Group2) \
  105. ((Group1) == (Group2) || \
  106. (Group2) == ScGlobalTDIGroup && (Group1) == ScGlobalPNP_TDIGroup)
  107. #define BELONGS_TO(Service, Group) \
  108. IS_SUBGROUP((Service)->MemberOfGroup, (Group))
  109. //-------------------------------------------------------------------//
  110. // //
  111. // Static global variables //
  112. // //
  113. //-------------------------------------------------------------------//
  114. //
  115. // For notifying us that a service has gone from start-pending state
  116. // to running or stopped state.
  117. //
  118. /* static */ HANDLE ScServiceChangeStateEvent = NULL;
  119. //
  120. // For serializing start requests
  121. //
  122. /* static */ CRITICAL_SECTION ScServiceStartCriticalSection;
  123. //
  124. // For telling internal routines when auto-start is in progress
  125. //
  126. BOOL ScAutoStartInProgress;
  127. //-------------------------------------------------------------------//
  128. // //
  129. // Local function prototypes //
  130. // //
  131. //-------------------------------------------------------------------//
  132. VOID
  133. ScSetServiceStartRequest(
  134. IN LPSERVICE_RECORD ServiceRecord,
  135. IN BOOL DemandStarting
  136. );
  137. BOOL
  138. ScMarkGroupStartNow(
  139. IN LPLOAD_ORDER_GROUP Group
  140. );
  141. DWORD
  142. ScGetCurrentGroupState(
  143. LPLOAD_ORDER_GROUP Group
  144. );
  145. DWORD
  146. ScStartMarkedServices(
  147. IN LPSERVICE_RECORD ServiceToStart OPTIONAL,
  148. IN DWORD NumArgs,
  149. IN LPSTRING_PTRSW CmdArgs,
  150. IN BOOL WaitForAll
  151. );
  152. VOID
  153. ScHandleServiceFailure(
  154. IN LPSERVICE_RECORD Service
  155. );
  156. BOOL
  157. ScDependenciesStarted(
  158. IN LPSERVICE_RECORD Service,
  159. OUT BOOL *IsBadDependencies,
  160. OUT BOOL *AllStarted,
  161. OUT BOOL *ExistsBlockedService
  162. );
  163. BOOL
  164. IsDependOnLaterGroup(
  165. IN LPLOAD_ORDER_GROUP ServiceGroup,
  166. IN LPLOAD_ORDER_GROUP DependOnGroup,
  167. IN DEPEND_TYPE DependType
  168. );
  169. VOID
  170. ScCleanupStartFailure(
  171. LPSERVICE_RECORD Service,
  172. DWORD StartError
  173. );
  174. VOID
  175. ScLookForHungServices(
  176. VOID
  177. );
  178. VOID
  179. ScHandleBadDependencies(
  180. VOID
  181. );
  182. BOOL
  183. ScServiceToStartDependOn(
  184. LPSERVICE_RECORD ServiceToStart OPTIONAL,
  185. LPSERVICE_RECORD StartPendingService
  186. );
  187. BOOL
  188. ScFoundDuplicateDependent(
  189. IN LPWSTR ServiceName,
  190. IN LPENUM_SERVICE_STATUS_WOW64 EnumBuffer,
  191. IN LPENUM_SERVICE_STATUS_WOW64 BufferEnd
  192. );
  193. #ifndef _CAIRO_
  194. VOID
  195. ScCheckNetLogonDepend(
  196. LPSERVICE_RECORD ServiceRecord,
  197. BOOL DemandStarting
  198. );
  199. #endif // _CAIRO_
  200. BOOL
  201. ScInitAutoStart(
  202. VOID
  203. )
  204. /*++
  205. Routine Description:
  206. This function creates the event for notifying the service controller
  207. that one of the automatically started service is running and creates
  208. the mutex for serializing start requests.
  209. Arguments:
  210. None.
  211. Return Value:
  212. TRUE - Event and mutex were created successfully. FALSE otherwise.
  213. --*/
  214. {
  215. //
  216. // Create event which indicates that some service that has been
  217. // automatically started is now running or stopped.
  218. //
  219. if ((ScServiceChangeStateEvent =
  220. CreateEvent(
  221. NULL, // Event attributes
  222. TRUE, // Event must be manually reset
  223. FALSE, // Initial state not signalled
  224. NULL
  225. )) == (HANDLE) NULL) {
  226. return FALSE;
  227. }
  228. //
  229. // Create critical section which is used to serialize start requests:
  230. // if auto-starting services, and a user tries to demand start
  231. // a service, auto-starting has to complete before we process the
  232. // demand start.
  233. //
  234. InitializeCriticalSection(&ScServiceStartCriticalSection);
  235. return TRUE;
  236. }
  237. DWORD
  238. ScAutoStartServices(
  239. IN OUT LPSC_RPC_LOCK lpLock
  240. )
  241. /*++
  242. Routine Description:
  243. This function automatically starts all services that must be
  244. auto-started, in group order.
  245. This routine may not return because we may instigate a reboot to
  246. revert to last-known-good.
  247. Arguments:
  248. None.
  249. Return Value:
  250. None.
  251. --*/
  252. {
  253. DWORD dwOOBEMode;
  254. ScAutoStartInProgress = TRUE;
  255. //
  256. // Set current request flag of all auto-start services as well as
  257. // their dependencies
  258. //
  259. SetupInProgress(NULL, &dwOOBEMode);
  260. if (dwOOBEMode)
  261. {
  262. #define NUM_OOBE_SERVICES 2
  263. DWORD dwError;
  264. DWORD dwTries = 0;
  265. LPSERVICE_RECORD ServiceRecord = NULL;
  266. HANDLE hPnPDone;
  267. HANDLE hMachineNameDone;
  268. LPWSTR lpServicesForOobe[NUM_OOBE_SERVICES] = { L"PlugPlay", L"AudioSrv" };
  269. UINT i;
  270. SC_LOG0(TRACE,
  271. "ScAutoStartServices: Is OOBE -- creating events\n");
  272. //
  273. // OOBE setup will wait on this event first. We set it
  274. // once PnP is done starting to wake up OOBE setup (which
  275. // will then prompt the user for the new machine name)
  276. //
  277. hPnPDone = CreateEvent(
  278. NULL, // Event Attributes
  279. TRUE, // ManualReset
  280. FALSE, // Initial State (not-signaled)
  281. SC_OOBE_PNP_DONE); // Name
  282. if (!hPnPDone)
  283. {
  284. dwError = GetLastError();
  285. SC_LOG1(ERROR,
  286. "ScAutoStartServices: Creation of SC_OOBE_PNP_DONE event failed %d\n",
  287. dwError);
  288. return dwError;
  289. }
  290. //
  291. // Once the machine name is set, OOBE setup will set this
  292. // event to tell us to finish auto-starting the services
  293. //
  294. hMachineNameDone = CreateEvent(
  295. NULL, // Event Attributes
  296. TRUE, // ManualReset
  297. FALSE, // Initial State (not-signaled)
  298. SC_OOBE_MACHINE_NAME_DONE); // Name
  299. if (!hMachineNameDone)
  300. {
  301. dwError = GetLastError();
  302. SC_LOG1(ERROR,
  303. "ScAutoStartServices: Creation of SC_OOBE_PNP_DONE event failed %d\n",
  304. dwError);
  305. CloseHandle(hPnPDone);
  306. return dwError;
  307. }
  308. SC_LOG0(TRACE,
  309. "ScAutoStartServices: Starting services used by OOBE\n");
  310. //
  311. // Start both services for OOBE, start only PlugPlay for SP install
  312. //
  313. for (i = 0; i < (dwOOBEMode != 2 ? NUM_OOBE_SERVICES : (UINT) 1); i++)
  314. {
  315. //
  316. // Since a computer name change may require PnP to be running,
  317. // start it and only it and wait until signalled to start the rest.
  318. // Grab the service list lock since ScGetNamedServiceRecord uses
  319. // the FOR_ALL_SERVICES macro, which expects it. Also start
  320. // AudioSrv, which is needed for audio for OOBE.
  321. //
  322. {
  323. CServiceListSharedLock LLock;
  324. dwError = ScGetNamedServiceRecord(lpServicesForOobe[i], &ServiceRecord);
  325. }
  326. if (dwError != NO_ERROR)
  327. {
  328. //
  329. // Ignore failure since ServiceRecord will now be NULL so
  330. // the call below will start all the auto-start services
  331. // instead of just PnP. The machine name will be wrong in
  332. // some of the services until the next reboot, but there's
  333. // nothing we can do about it at this point.
  334. //
  335. SC_LOG2(ERROR,
  336. "ScAutoStartServices: OOBE lookup of service %ws failed %d\n",
  337. lpServicesForOobe[i],
  338. dwError);
  339. }
  340. dwError = ScStartServiceAndDependencies(ServiceRecord, 0, NULL, TRUE);
  341. if (dwError != NO_ERROR)
  342. {
  343. //
  344. // Ignore failure -- it's possible PnP won't need to be
  345. // up and running (it's only necessary if the OEM setup
  346. // for the machine was done without a keyboard and PnP
  347. // is now needed to detect the hardware). Audio failure
  348. // is definitely non-fatal.
  349. //
  350. SC_LOG2(ERROR,
  351. "ScAutoStartServices: OOBE start of %ws failed %d\n",
  352. lpServicesForOobe[i],
  353. dwError);
  354. }
  355. }
  356. //
  357. // Drop the database lock since OOBE may invoke server-side PnP
  358. // installations and setupapi.dll will try to lock the database.
  359. // In addition, it may invoke 3rd-party class installers and we
  360. // have no idea what they may try to do with the lock.
  361. //
  362. ScUnlockDatabase(lpLock);
  363. //
  364. // Let setup proceed
  365. //
  366. if (!SetEvent(hPnPDone))
  367. {
  368. dwError = GetLastError();
  369. SC_LOG1(ERROR,
  370. "ScAutoStartServices: Failed to set event (hPnPDone) %d\n",
  371. dwError);
  372. }
  373. //
  374. // Wait for setup to let us proceed.
  375. //
  376. WaitForSingleObject(hMachineNameDone, INFINITE);
  377. dwError = NO_ERROR;
  378. //
  379. // Reacquire the database lock. Wait 10 minutes at most.
  380. //
  381. do
  382. {
  383. if (dwError != NO_ERROR)
  384. {
  385. Sleep(2000);
  386. }
  387. dwError = ScLockDatabase(TRUE, SERVICES_ACTIVE_DATABASEW, lpLock);
  388. dwTries++;
  389. }
  390. while (dwError != NO_ERROR && dwTries != 300);
  391. CloseHandle(hPnPDone);
  392. CloseHandle(hMachineNameDone);
  393. if (dwError != NO_ERROR)
  394. {
  395. SC_LOG1(ERROR,
  396. "ScAutoStartServices: Failed to reacquire database lock %d\n",
  397. dwError);
  398. return dwError;
  399. }
  400. #undef NUM_OOBE_SERVICES
  401. }
  402. //
  403. // Start services with start request flag set in group order
  404. //
  405. ScStartServiceAndDependencies(NULL, 0, NULL, FALSE);
  406. ScAutoStartInProgress = FALSE;
  407. return NO_ERROR;
  408. }
  409. DWORD
  410. ScStartServiceAndDependencies(
  411. IN LPSERVICE_RECORD ServiceToStart OPTIONAL,
  412. IN DWORD NumArgs,
  413. IN LPSTRING_PTRSW CmdArgs,
  414. IN BOOL fIsOOBE
  415. )
  416. /*++
  417. Routine Description:
  418. This function marks a group or a service to be started now before
  419. calling ScStartMarkedServices to start them.
  420. Arguments:
  421. ServiceToStart - Service to be started; or, NULL if autostarting
  422. services.
  423. NumArgs, CmdArgs - Arguments for the service to be started.
  424. Return Value:
  425. NO_ERROR if successful; otherwise, the return value from the first
  426. unsuccessful call to ScStartMarkedServices.
  427. --*/
  428. {
  429. DWORD status;
  430. DWORD ApiStatus = NO_ERROR;
  431. PLOAD_ORDER_GROUP Group;
  432. SC_RPC_LOCK Lock;
  433. BOOL databaseLocked = FALSE;
  434. //
  435. // Serialize start requests by allowing auto-starting of groups or
  436. // demand start of a service one at a time.
  437. //
  438. EnterCriticalSection(&ScServiceStartCriticalSection);
  439. //
  440. // Grab the SC Manager database lock.
  441. //
  442. if (!ScStillInitializing) {
  443. if ((status = ScLockDatabase(
  444. TRUE, // called internally
  445. SERVICES_ACTIVE_DATABASEW,
  446. &Lock
  447. )) != NO_ERROR) {
  448. LeaveCriticalSection(&ScServiceStartCriticalSection);
  449. return status;
  450. }
  451. databaseLocked = TRUE;
  452. }
  453. //
  454. // There is one (highly unlikely) condition in which we might need to
  455. // modify the group information here: if we call ScStartService and it
  456. // calls ScDeactivateSR on a service whose update flag is set (its
  457. // config was changed while it was running).
  458. // To prevent deadlocks, the grouplist lock is always acquired before
  459. // the service database lock, if both are needed. So we get an
  460. // exclusive lock on the grouplist here.
  461. //
  462. {
  463. CGroupListExclusiveLock GLock;
  464. //
  465. // Get a shared lock on the service list as we shall walk it several
  466. // times.
  467. //
  468. CServiceListSharedLock LLock;
  469. //
  470. // Get the exclusive database lock so that we can increment the
  471. // use count of the service being started as well as their dependencies
  472. // because otherwise they could go away if deleted.
  473. //
  474. {
  475. CServiceRecordExclusiveLock RLock;
  476. if (ARGUMENT_PRESENT(ServiceToStart)) {
  477. //
  478. // Demand starting a service.
  479. //
  480. //
  481. // We can never start a disabled service
  482. //
  483. if (ServiceToStart->StartType == SERVICE_DISABLED ||
  484. ! ScInHardwareProfile(ServiceToStart->ServiceName, 0)) {
  485. ApiStatus = ERROR_SERVICE_DISABLED;
  486. goto ReleaseLocks;
  487. }
  488. //
  489. // Cannot start a deleted service.
  490. //
  491. if (DELETE_FLAG_IS_SET(ServiceToStart)) {
  492. ApiStatus = ERROR_SERVICE_MARKED_FOR_DELETE;
  493. goto ReleaseLocks;
  494. }
  495. //
  496. // Get the current state of the service
  497. //
  498. if (ServiceToStart->ServiceStatus.dwServiceType & SERVICE_DRIVER) {
  499. (void) ScGetDriverStatus(ServiceToStart, NULL);
  500. }
  501. if (ServiceToStart->ServiceStatus.dwCurrentState != SERVICE_STOPPED) {
  502. ApiStatus = ERROR_SERVICE_ALREADY_RUNNING;
  503. goto ReleaseLocks;
  504. }
  505. //
  506. // Set the current-start flag on the service and all of
  507. // its dependencies. Also increment their use counts so
  508. // that we know they won't go away, even when we release
  509. // the locks temporarily.
  510. //
  511. SC_LOG(DEPEND,"Setting start request for %ws and dependencies\n",
  512. ServiceToStart->ServiceName);
  513. SC_LOG(WHY, " Will start %ws because of an explicit start request\n",
  514. ServiceToStart->ServiceName);
  515. ScSetServiceStartRequest(ServiceToStart, TRUE);
  516. }
  517. else {
  518. //
  519. // Auto-starting services.
  520. //
  521. // If we are in GUI mode setup, we only start those auto-start
  522. // services that are allowed. Allowed services are listed
  523. // under HKLM\System\Setup\AllowStart as registry keys.
  524. // If the subkey with the same name as the service exits,
  525. // it means we allow it to be autostarted during GUI mode.
  526. //
  527. HKEY hkeyAllowStart = NULL;
  528. BOOL InSetup = SetupInProgress(NULL, NULL);
  529. if (InSetup)
  530. {
  531. status = ScRegOpenKeyExW(
  532. HKEY_LOCAL_MACHINE,
  533. REG_KEY_SETUP_ALLOW_START,
  534. REG_OPTION_NON_VOLATILE,
  535. KEY_READ,
  536. &hkeyAllowStart);
  537. if (NO_ERROR != status)
  538. {
  539. // This is not expected as the registry key is created
  540. // by hivesys.inf. We'll skip checking services in
  541. // this case.
  542. //
  543. hkeyAllowStart = NULL;
  544. }
  545. }
  546. // Set the CurrentStartRequest flag to TRUE for all services
  547. // of type AUTO_START that are enabled in this hardware profile,
  548. // and their dependencies.
  549. //
  550. SC_LOG0(DEPEND,"Setting start request for auto-started services and dependencies\n");
  551. FOR_SERVICES_THAT(Service, Service->StartType == SERVICE_AUTO_START &&
  552. ScInHardwareProfile(Service->ServiceName, 0))
  553. {
  554. BOOL AllowStart = TRUE;
  555. if (hkeyAllowStart)
  556. {
  557. HKEY hkey;
  558. ASSERT(InSetup);
  559. status = ScRegOpenKeyExW(
  560. hkeyAllowStart,
  561. Service->ServiceName,
  562. REG_OPTION_NON_VOLATILE,
  563. KEY_READ,
  564. &hkey);
  565. if (NO_ERROR == status)
  566. {
  567. // The service name exists and is therefore
  568. // allowed to start. Don't forget to close the
  569. // key.
  570. ScRegCloseKey(hkey);
  571. ASSERT(AllowStart);
  572. }
  573. else
  574. {
  575. AllowStart = FALSE;
  576. SC_LOG(WHY, " Will NOT start %ws because it's not "
  577. "allowed to run during system setup\n",
  578. Service->ServiceName);
  579. }
  580. }
  581. if (AllowStart)
  582. {
  583. SC_LOG(WHY, " Will start %ws because it's configured to autostart\n",
  584. Service->ServiceName);
  585. ScSetServiceStartRequest(Service, FALSE);
  586. }
  587. }
  588. if (hkeyAllowStart)
  589. {
  590. ScRegCloseKey(hkeyAllowStart);
  591. }
  592. }
  593. } // Release RLock
  594. //
  595. // Always start services in group order.
  596. //
  597. for (Group = ScGetOrderGroupList();
  598. Group != NULL;
  599. Group = Group->Next)
  600. {
  601. //
  602. // Start each group in load group order
  603. //
  604. if (ScMarkGroupStartNow(Group)) {
  605. SC_LOG(DEPEND,"------ Starting services in group \"%ws\" -----\n", Group->GroupName);
  606. BOOL WaitForGroup;
  607. if (ARGUMENT_PRESENT(ServiceToStart) &&
  608. ServiceToStart->MemberOfGroup == Group) {
  609. //
  610. // Don't have to wait for all marked members of the group
  611. // to finish starting because the service which is demand
  612. // started is polled by the UI.
  613. //
  614. WaitForGroup = FALSE;
  615. }
  616. else {
  617. //
  618. // Auto-starting (ServiceToStart == NULL) or demand-starting
  619. // a service that is not within this group. Wait for group
  620. // all marked members finish starting.
  621. //
  622. WaitForGroup = TRUE;
  623. }
  624. status = ScStartMarkedServices(
  625. ServiceToStart,
  626. NumArgs,
  627. CmdArgs,
  628. (WaitForGroup || fIsOOBE)
  629. );
  630. if (status != NO_ERROR && ApiStatus == NO_ERROR) {
  631. //
  632. // Save first error to be returned
  633. //
  634. ApiStatus = status;
  635. }
  636. }
  637. }
  638. //
  639. // Services that do not belong in any group are considered
  640. // in a group that starts last.
  641. //
  642. if (ScMarkGroupStartNow(NULL)) {
  643. SC_LOG0(DEPEND,"------ Starting services that aren't in ServiceGroupOrder -----\n");
  644. //
  645. // Wait only for auto-start services or for the service(s)
  646. // we explicitly demand-start during OOBE setup
  647. //
  648. status = ScStartMarkedServices(
  649. ServiceToStart,
  650. NumArgs,
  651. CmdArgs,
  652. (! ARGUMENT_PRESENT(ServiceToStart) || fIsOOBE)
  653. );
  654. if (status != NO_ERROR && ApiStatus == NO_ERROR) {
  655. //
  656. // Save first error to be returned
  657. //
  658. ApiStatus = status;
  659. }
  660. }
  661. SC_LOG(DEPEND,"------ Done %s-starting services -----\n",
  662. ARGUMENT_PRESENT(ServiceToStart) ? "demand" : "auto");
  663. //
  664. // Clear the CurrentStartRequest flags when done starting service(s).
  665. //
  666. {
  667. CServiceRecordExclusiveLock RLock;
  668. FOR_SERVICES_THAT(Service, CURRENTSTART_FLAG_IS_SET(Service))
  669. {
  670. CLEAR_CURRENTSTART_FLAG(Service);
  671. ScDecrementUseCountAndDelete(Service);
  672. }
  673. }
  674. } // Release LLock and GLock
  675. ReleaseLocks:
  676. //
  677. // Release the SC Manager database lock.
  678. //
  679. if (databaseLocked) {
  680. ScUnlockDatabase(&Lock);
  681. }
  682. LeaveCriticalSection(&ScServiceStartCriticalSection);
  683. return ApiStatus;
  684. }
  685. VOID
  686. ScSetServiceStartRequest(
  687. IN LPSERVICE_RECORD ServiceRecord,
  688. IN BOOL DemandStarting
  689. )
  690. /*++
  691. Routine Description:
  692. This function sets the CurrentStartRequest flag of the specified service
  693. to TRUE and recursively sets the flag of all the services this
  694. service depends on. It also initializes the StartState and StartError
  695. of the services that are about to be started.
  696. Arguments:
  697. ServiceRecord - Supplies a pointer to the service record of service
  698. to be started.
  699. DemandStarting - Supplies a flag that is set to TRUE if we are demand-
  700. starting a service, FALSE if we are auto-starting services.
  701. Return Value:
  702. None.
  703. Note:
  704. This function expects the caller to have held the exclusive service
  705. database lock. This function is called by ScStartServiceAndDependencies.
  706. --*/
  707. {
  708. SC_ASSERT(ScServiceListLock.Have());
  709. SC_ASSERT(ScServiceRecordLock.HaveExclusive());
  710. if (CURRENTSTART_FLAG_IS_SET(ServiceRecord)) {
  711. return;
  712. }
  713. //
  714. // Set the CurrentStartRequest to TRUE
  715. //
  716. SET_CURRENTSTART_FLAG(ServiceRecord);
  717. //
  718. // Update the StartState and StartError
  719. //
  720. if (ServiceRecord->StartType == SERVICE_DISABLED ||
  721. ! ScInHardwareProfile(ServiceRecord->ServiceName, 0)) {
  722. ServiceRecord->StartState = SC_START_FAIL;
  723. ServiceRecord->StartError = ERROR_SERVICE_DISABLED;
  724. }
  725. else if (DELETE_FLAG_IS_SET(ServiceRecord)) {
  726. ServiceRecord->StartState = SC_START_FAIL;
  727. ServiceRecord->StartError = ERROR_SERVICE_MARKED_FOR_DELETE;
  728. }
  729. else {
  730. if (ServiceRecord->ServiceStatus.dwServiceType & SERVICE_DRIVER) {
  731. (void) ScGetDriverStatus(ServiceRecord, NULL);
  732. }
  733. switch (ServiceRecord->ServiceStatus.dwCurrentState) {
  734. case SERVICE_STOPPED:
  735. if (DemandStarting) {
  736. //
  737. // Demand starting a service. We want to retry
  738. // eventhough we have failed once before.
  739. //
  740. ServiceRecord->StartState = SC_NEVER_STARTED;
  741. }
  742. else {
  743. //
  744. // Auto-starting bunch of services at boot. If
  745. // the service was ever started before and failed,
  746. // we don't want to start it again.
  747. //
  748. if (ServiceRecord->ServiceStatus.dwWin32ExitCode !=
  749. ERROR_SERVICE_NEVER_STARTED) {
  750. ServiceRecord->StartState = SC_START_FAIL;
  751. }
  752. else {
  753. ServiceRecord->StartState = SC_NEVER_STARTED;
  754. }
  755. }
  756. break;
  757. case SERVICE_START_PENDING:
  758. ServiceRecord->StartState = SC_START_PENDING;
  759. break;
  760. case SERVICE_STOP_PENDING:
  761. case SERVICE_PAUSED:
  762. case SERVICE_CONTINUE_PENDING:
  763. case SERVICE_PAUSE_PENDING:
  764. case SERVICE_RUNNING:
  765. ServiceRecord->StartState = SC_START_SUCCESS;
  766. break;
  767. default:
  768. SC_LOG1(
  769. ERROR,
  770. "ScSetServiceStartRequest: Unexpected dwCurrentState %0lx\n",
  771. ServiceRecord->ServiceStatus.dwCurrentState
  772. );
  773. SC_ASSERT(FALSE);
  774. ServiceRecord->StartState = SC_START_FAIL;
  775. break;
  776. }
  777. }
  778. //
  779. // Increment the reference count so that the dependency service
  780. // never goes away while we are in the process of starting them.
  781. //
  782. ServiceRecord->UseCount++;
  783. SC_LOG2(USECOUNT, "ScSetServiceStartRequest: " FORMAT_LPWSTR
  784. " increment USECOUNT=%lu\n", ServiceRecord->ServiceName, ServiceRecord->UseCount);
  785. SC_LOG2(DEPEND_DUMP, "CSR=TRUE for "
  786. FORMAT_LPWSTR " USECOUNT=%lu\n", ServiceRecord->ServiceName,
  787. ServiceRecord->UseCount);
  788. //
  789. // For each of this service's dependencies
  790. //
  791. for (LPDEPEND_RECORD Depend = ServiceRecord->StartDepend;
  792. Depend != NULL;
  793. Depend = Depend->Next)
  794. {
  795. if (Depend->DependType == TypeDependOnService) {
  796. if (CURRENTSTART_FLAG_IS_SET(Depend->DependService)) {
  797. //
  798. // CurrentStartRequest of a dependency service is already
  799. // set to TRUE. Just go on to next dependency.
  800. //
  801. SC_LOG2(WHY, " (Will start %ws because %ws depends on it too)\n",
  802. Depend->DependService->ServiceName, ServiceRecord->ServiceName);
  803. SC_LOG2(DEPEND_DUMP, "DependService " FORMAT_LPWSTR
  804. " CSR=TRUE already, USECOUNT=%lu\n",
  805. Depend->DependService->ServiceName,
  806. Depend->DependService->UseCount);
  807. }
  808. else {
  809. SC_LOG2(WHY, " Will start %ws because %ws depends on it\n",
  810. Depend->DependService->ServiceName, ServiceRecord->ServiceName);
  811. ScSetServiceStartRequest(Depend->DependService, DemandStarting);
  812. }
  813. }
  814. else if (Depend->DependType == TypeDependOnGroup) {
  815. //
  816. // This service has a dependency on a group.
  817. // For each service in that group
  818. //
  819. FOR_SERVICES_THAT(Service, BELONGS_TO(Service, Depend->DependGroup))
  820. {
  821. if (CURRENTSTART_FLAG_IS_SET(Service)) {
  822. //
  823. // CurrentStartRequest of a dependency service is
  824. // already set to TRUE. Just go on to next dependency.
  825. //
  826. SC_LOG3(WHY, " (Will start %ws because %ws depends on its group %ws too)\n",
  827. Service->ServiceName, ServiceRecord->ServiceName,
  828. Depend->DependGroup->GroupName);
  829. SC_LOG3(DEPEND_DUMP, "DependGroup " FORMAT_LPWSTR
  830. ", Service " FORMAT_LPWSTR
  831. " CSR=TRUE already, USECOUNT=%lu\n",
  832. Depend->DependGroup->GroupName,
  833. Service->ServiceName, Service->UseCount);
  834. }
  835. else {
  836. SC_LOG3(WHY, " Will start %ws because %ws depends on its group %ws\n",
  837. Service->ServiceName, ServiceRecord->ServiceName,
  838. Depend->DependGroup->GroupName);
  839. ScSetServiceStartRequest(
  840. Service,
  841. DemandStarting
  842. );
  843. }
  844. }
  845. }
  846. }
  847. #ifndef _CAIRO_
  848. //
  849. // We have now gone through all the dependencies that are listed. Now
  850. // Determine if this service needs to depend on NetLogon. If the service
  851. // runs in an account, it may require NetLogon.
  852. //
  853. ScCheckNetLogonDepend(ServiceRecord,DemandStarting);
  854. #endif // _CAIRO_
  855. }
  856. BOOL
  857. ScMarkGroupStartNow(
  858. IN LPLOAD_ORDER_GROUP Group
  859. )
  860. /*++
  861. Routine Description:
  862. This function go through all services that belong in the specified
  863. group and mark the services that have the CurrentStartRequest flag
  864. set to be started immediately.
  865. Arguments:
  866. Group - Supplies a pointer to the load order group to mark for
  867. start.
  868. Return Value:
  869. Returns TRUE if at least one member of the group is marked
  870. START_NOW or is START_PENDING. FALSE otherwise. This flag
  871. is to indicate whether ScStartMarkedServices should be called
  872. to handle starting a group.
  873. --*/
  874. {
  875. BOOL ReturnFlag = FALSE;
  876. //
  877. // Mark all the CurrentStartRequest (which includes all auto-start)
  878. // services to be started now
  879. //
  880. // A service is marked START_NOW if it is a member of the specified
  881. // group. If the specified group is NULL, mark all services that
  882. // do not belong to any group as well as services that belong to
  883. // standalone groups.
  884. //
  885. FOR_SERVICES_THAT(Service,
  886. ((Service->MemberOfGroup == Group) ||
  887. (Group == NULL && (Service->MemberOfGroup != NULL) &&
  888. (Service->MemberOfGroup->RefCount != MAXULONG) ))
  889. &&
  890. CURRENTSTART_FLAG_IS_SET(Service) )
  891. {
  892. if (Service->StartState == SC_NEVER_STARTED)
  893. {
  894. Service->StartState = SC_START_NOW;
  895. Service->StartError = NO_ERROR;
  896. }
  897. if (Service->StartState == SC_START_NOW ||
  898. Service->StartState == SC_START_PENDING)
  899. {
  900. ReturnFlag = TRUE;
  901. }
  902. }
  903. return ReturnFlag;
  904. }
  905. DWORD
  906. RI_ScGetCurrentGroupStateW(
  907. IN SC_RPC_HANDLE hSCManager,
  908. IN LPWSTR pszGroupName,
  909. OUT LPDWORD pdwCurrentState
  910. )
  911. /*++
  912. Routine Description:
  913. This is obsolete, but the RPC entry point still exists.
  914. --*/
  915. {
  916. return ERROR_NOT_SUPPORTED;
  917. }
  918. DWORD
  919. ScGetCurrentGroupState(
  920. LPLOAD_ORDER_GROUP Group
  921. )
  922. {
  923. BOOL OneStarted = FALSE;
  924. FOR_SERVICES_THAT(Service, BELONGS_TO(Service, Group))
  925. {
  926. switch (Service->StartState)
  927. {
  928. case SC_NEVER_STARTED:
  929. //
  930. // Ignore services that are disabled or marked for
  931. // deletion.
  932. // (This check is really needed only when this function is
  933. // called from RI_ScGetCurrentGroupState. When called
  934. // from ScDependenciesStarted, such services will already
  935. // have had their StartState set to SC_START_FAIL.)
  936. //
  937. if (Service->StartType == SERVICE_DISABLED ||
  938. DELETE_FLAG_IS_SET(Service) ||
  939. ! ScInHardwareProfile(Service->ServiceName, 0))
  940. {
  941. continue;
  942. }
  943. //
  944. // else fall through
  945. //
  946. case SC_START_NOW:
  947. case SC_START_PENDING:
  948. SC_LOG2(DEPEND, "Group " FORMAT_LPWSTR " NOT started "
  949. "because of Service " FORMAT_LPWSTR "\n",
  950. Group->GroupName, Service->ServiceName);
  951. return GROUP_NOT_STARTED;
  952. case SC_START_SUCCESS:
  953. OneStarted = TRUE;
  954. break; // out of switch, not out of loop
  955. }
  956. }
  957. if (OneStarted)
  958. {
  959. SC_LOG1(DEPEND, "Group " FORMAT_LPWSTR " ONE started\n",
  960. Group->GroupName);
  961. return GROUP_ONE_STARTED;
  962. }
  963. else
  964. {
  965. SC_LOG1(DEPEND, "Group " FORMAT_LPWSTR " FAILED to start\n",
  966. Group->GroupName);
  967. return GROUP_START_FAIL;
  968. }
  969. }
  970. DWORD
  971. ScStartMarkedServices(
  972. IN LPSERVICE_RECORD ServiceToStart OPTIONAL,
  973. IN DWORD NumArgs,
  974. IN LPSTRING_PTRSW CmdArgs,
  975. IN BOOL WaitForAll
  976. )
  977. /*++
  978. Routine Description:
  979. This function starts the services that are marked as SERVICE_START_NOW
  980. in the service record list. Once the service is running, or if the
  981. service failed to start, the SERVICE_START_NOW bit is removed.
  982. If a service marked as SERVICE_START_NOW depends on a service that is
  983. not marked, the dependency service will also be marked SERVICE_START_NOW.
  984. Arguments:
  985. ServiceToStart - Supplies a pointer to the service which is to be demand
  986. started via the StartService API. If this parameter is NULL, this
  987. routine is called by the service controller to auto-start services
  988. at boot.
  989. NumArgs - Supplies the number of command-line arguments for the demand
  990. started service. If ServiceToStart is NULL, this parameter is ignored.
  991. CmdArgs - Supplies an array of command arguments to the demand started
  992. service. If ServiceToStart is NULL, this parameter is ignored.
  993. WaitForAll - Supplies a flag which if TRUE tells this function to
  994. wait until all start-pending services that were marked START_NOW
  995. to get done.
  996. Return Value:
  997. Returns error if failure to reset the ScServiceChangeStateEvent.
  998. --*/
  999. {
  1000. DWORD Error;
  1001. BOOL AllStarted;
  1002. BOOL ExistsBlockedService;
  1003. BOOL IsBadDependencies;
  1004. BOOL IsStartPending;
  1005. DWORD ServiceCurrentState;
  1006. #if DBG
  1007. DWORD LoopCount = 0;
  1008. #endif
  1009. //
  1010. // Reset ScServiceChangeStateEvent to non-signalled state
  1011. //
  1012. if (! ResetEvent(ScServiceChangeStateEvent)) {
  1013. Error = GetLastError();
  1014. //
  1015. // This is a serious error--we cannot proceed.
  1016. //
  1017. SC_LOG1(ERROR, "Error reseting ScServiceChangeStateEvent " FORMAT_DWORD
  1018. "\n", Error);
  1019. ScLogEvent(
  1020. NEVENT_CALL_TO_FUNCTION_FAILED,
  1021. SC_RESET_EVENT,
  1022. Error
  1023. );
  1024. return Error;
  1025. }
  1026. //
  1027. // Start all services that are marked
  1028. //
  1029. do { // while (! AllStarted)
  1030. AllStarted = TRUE;
  1031. IsStartPending = FALSE;
  1032. ExistsBlockedService = FALSE;
  1033. IsBadDependencies = FALSE;
  1034. SC_LOG1(DEPEND, "BIG LOOP COUNT " FORMAT_DWORD "\n", LoopCount++);
  1035. //
  1036. // Loop through every service which is currently in the database
  1037. //
  1038. FOR_ALL_SERVICES(Service)
  1039. {
  1040. //
  1041. // Check if the current service failed to start, and if we have
  1042. // to revert to last-known-good. Don't revert if demand start.
  1043. //
  1044. if (! ARGUMENT_PRESENT(ServiceToStart)) {
  1045. ScHandleServiceFailure(Service);
  1046. }
  1047. if (Service->StartState == SC_START_NOW) {
  1048. SC_LOG1(DEPEND, FORMAT_LPWSTR " is marked START NOW\n",
  1049. Service->ServiceName);
  1050. //
  1051. // Start the current service only if all its dependencies
  1052. // have started successfully.
  1053. //
  1054. if (ScDependenciesStarted(
  1055. Service,
  1056. &IsBadDependencies,
  1057. &AllStarted,
  1058. &ExistsBlockedService
  1059. )) {
  1060. //
  1061. // Start the service and save the start error code
  1062. //
  1063. SC_LOG1(DEPEND, "ScStartMarkedServices: Starting "
  1064. FORMAT_LPWSTR "\n", Service->ServiceName);
  1065. if (Service == ServiceToStart)
  1066. {
  1067. Service->StartError = ScStartService(
  1068. Service,
  1069. NumArgs,
  1070. CmdArgs
  1071. );
  1072. }
  1073. else
  1074. {
  1075. Service->StartError = ScStartService(
  1076. Service,
  1077. 0,
  1078. NULL
  1079. );
  1080. //
  1081. // We are starting a new service so remember to loop
  1082. // through again to process any service which are
  1083. // dependent on it. Don't have to set AllStarted
  1084. // to FALSE if this service is ServiceToStart because
  1085. // nothing is dependent on it since it is demand
  1086. // started.
  1087. //
  1088. AllStarted = FALSE;
  1089. }
  1090. if (Service->StartError == NO_ERROR ||
  1091. Service->StartError == ERROR_SERVICE_ALREADY_RUNNING)
  1092. {
  1093. //
  1094. // Get the state of the just started service
  1095. //
  1096. {
  1097. CServiceRecordSharedLock RLock;
  1098. ServiceCurrentState =
  1099. Service->ServiceStatus.dwCurrentState;
  1100. }
  1101. switch (ServiceCurrentState) {
  1102. case SERVICE_START_PENDING:
  1103. IsStartPending = TRUE;
  1104. Service->StartState = SC_START_PENDING;
  1105. break;
  1106. case SERVICE_STOP_PENDING:
  1107. case SERVICE_PAUSED:
  1108. case SERVICE_CONTINUE_PENDING:
  1109. case SERVICE_PAUSE_PENDING:
  1110. case SERVICE_RUNNING:
  1111. Service->StartState = SC_START_SUCCESS;
  1112. break;
  1113. case SERVICE_STOPPED:
  1114. Service->StartState = SC_START_FAIL;
  1115. break;
  1116. default:
  1117. SC_LOG1(ERROR, "Unexpected service state "
  1118. FORMAT_HEX_DWORD "\n",
  1119. ServiceCurrentState);
  1120. SC_ASSERT(FALSE);
  1121. Service->StartState = SC_START_FAIL;
  1122. }
  1123. }
  1124. else {
  1125. //
  1126. // Clear ERROR_SERVICE_NEVER_STARTED in the Win32ExitCode
  1127. // field if service failed to start.
  1128. //
  1129. {
  1130. CServiceRecordExclusiveLock RLock;
  1131. if (Service->ServiceStatus.dwWin32ExitCode ==
  1132. ERROR_SERVICE_NEVER_STARTED) {
  1133. Service->ServiceStatus.dwWin32ExitCode =
  1134. Service->StartError;
  1135. }
  1136. }
  1137. Service->StartState = SC_START_FAIL;
  1138. //
  1139. // For popup after user has logged on to indicate that some
  1140. // service started at boot has failed.
  1141. // We don't log the error if it is ERROR_IGNORE or if
  1142. // we tried to start a non-Safeboot service in Safeboot.
  1143. //
  1144. if (Service->ErrorControl != SERVICE_ERROR_IGNORE
  1145. &&
  1146. Service->StartError != ERROR_NOT_SAFEBOOT_SERVICE)
  1147. {
  1148. ScLogEvent(
  1149. NEVENT_SERVICE_START_FAILED,
  1150. Service->DisplayName,
  1151. Service->StartError
  1152. );
  1153. ScPopupStartFail = TRUE;
  1154. }
  1155. }
  1156. }
  1157. }
  1158. else if (Service->StartState == SC_START_PENDING) {
  1159. //
  1160. // We need to wait for this pending service to be completely
  1161. // started if:
  1162. // 1) We are auto-starting services in sequence;
  1163. // ServiceToStart == NULL
  1164. // 2) We are demand starting ServiceToStart and
  1165. // it depends on services that are currently
  1166. // start-pending
  1167. //
  1168. // We don't wait if the pending service is started by demand
  1169. // and is unrelated in the start sequence of ServiceToStart,
  1170. // or it is ServiceToStart itself.
  1171. //
  1172. if ((Service != ServiceToStart) &&
  1173. ScServiceToStartDependOn(ServiceToStart, Service))
  1174. {
  1175. SC_LOG3(DEPEND, FORMAT_LPWSTR " is still PENDING "
  1176. "(chkpt %lu, wait hint %lu ms)\n",
  1177. Service->ServiceName,
  1178. Service->ServiceStatus.dwCheckPoint,
  1179. Service->ServiceStatus.dwWaitHint);
  1180. //
  1181. // If a service makes some hokey state changes, it will be
  1182. // stuck in the SC_START_PENDING state, which means that we'll
  1183. // loop/wait forever. Correct things if the service is in a
  1184. // "confused" state. Note that this can happen for a service
  1185. // that goes from SERVICE_START_PENDING to anything other
  1186. // than SERVICE_RUNNING, SERVICE_STOPPED, or SERVICE_STOP_PENDING
  1187. // (and can then change to one of those three states without
  1188. // causing the state-change event to be set)
  1189. //
  1190. //
  1191. // Get the state of the pending service
  1192. //
  1193. {
  1194. CServiceRecordSharedLock RLock;
  1195. ServiceCurrentState = Service->ServiceStatus.dwCurrentState;
  1196. }
  1197. switch (ServiceCurrentState)
  1198. {
  1199. case SERVICE_START_PENDING:
  1200. IsStartPending = TRUE;
  1201. ExistsBlockedService = TRUE;
  1202. AllStarted = FALSE;
  1203. break;
  1204. case SERVICE_STOP_PENDING:
  1205. case SERVICE_PAUSED:
  1206. case SERVICE_CONTINUE_PENDING:
  1207. case SERVICE_PAUSE_PENDING:
  1208. case SERVICE_RUNNING:
  1209. Service->StartState = SC_START_SUCCESS;
  1210. break;
  1211. case SERVICE_STOPPED:
  1212. Service->StartState = SC_START_FAIL;
  1213. break;
  1214. default:
  1215. SC_LOG1(ERROR,
  1216. "Unexpected service state %x\n",
  1217. ServiceCurrentState);
  1218. SC_ASSERT(FALSE);
  1219. Service->StartState = SC_START_FAIL;
  1220. }
  1221. }
  1222. }
  1223. } // for every service
  1224. //
  1225. // Only wait for services to finish starting if:
  1226. // the services are auto-started at boot
  1227. // the services are required to be running before a service that
  1228. // is demand-started can run.
  1229. //
  1230. if (IsStartPending && (ExistsBlockedService || WaitForAll))
  1231. {
  1232. SC_LOG0(DEPEND, "About to wait on ScServiceChangeEvent\n");
  1233. //
  1234. // ScServiceChangeStateEvent is signalled by RSetServiceStatus whenever
  1235. // a service changes its state from SERVICE_START_PENDING to
  1236. // SERVICE_RUNNING or SERVICE_STOPPED.
  1237. //
  1238. Error = WaitForSingleObject(
  1239. ScServiceChangeStateEvent,
  1240. SERVICE_START_TIMEOUT
  1241. );
  1242. if (Error == WAIT_TIMEOUT)
  1243. {
  1244. //
  1245. // Go through all services and see if any one has hung
  1246. // while starting.
  1247. //
  1248. ScLookForHungServices();
  1249. }
  1250. else if (Error == 0)
  1251. {
  1252. //
  1253. // Reset ScServiceChangeStateEvent to non-signalled state
  1254. //
  1255. if (!ResetEvent(ScServiceChangeStateEvent))
  1256. {
  1257. Error = GetLastError();
  1258. //
  1259. // This is a serious error--we cannot proceed.
  1260. //
  1261. SC_LOG1(ERROR, "Error reseting ScServiceChangeStateEvent "
  1262. FORMAT_DWORD "\n", Error);
  1263. ScLogEvent(
  1264. NEVENT_CALL_TO_FUNCTION_FAILED,
  1265. SC_RESET_EVENT,
  1266. Error
  1267. );
  1268. return Error;
  1269. }
  1270. }
  1271. else if (Error == 0xffffffff)
  1272. {
  1273. //
  1274. // An error has occurred
  1275. //
  1276. SC_LOG1(ERROR,
  1277. "Wait for ScServiceChangeStateEvent returned %d\n",
  1278. GetLastError());
  1279. SC_ASSERT(FALSE);
  1280. }
  1281. }
  1282. else if ((AllStarted && ExistsBlockedService) || IsBadDependencies)
  1283. {
  1284. //
  1285. // Circular dependencies!
  1286. //
  1287. SC_LOG0(ERROR, "Detected circular dependencies!!\n");
  1288. SC_LOG3(ERROR,
  1289. "AllStarted=" FORMAT_DWORD
  1290. ", ExistsBlockedService=" FORMAT_DWORD
  1291. ", IsBadDependencies=" FORMAT_DWORD "\n",
  1292. (DWORD) AllStarted, (DWORD) ExistsBlockedService,
  1293. (DWORD) IsBadDependencies);
  1294. if (ARGUMENT_PRESENT(ServiceToStart))
  1295. {
  1296. SC_LOG1(ERROR, " Demand starting " FORMAT_LPWSTR "\n",
  1297. ServiceToStart->DisplayName);
  1298. ScLogEvent(
  1299. NEVENT_CIRCULAR_DEPENDENCY_DEMAND,
  1300. ServiceToStart->DisplayName
  1301. );
  1302. }
  1303. else
  1304. {
  1305. SC_LOG0(ERROR, " Auto-starting services\n");
  1306. ScLogEvent(NEVENT_CIRCULAR_DEPENDENCY_AUTO);
  1307. ScHandleBadDependencies();
  1308. }
  1309. return ERROR_CIRCULAR_DEPENDENCY;
  1310. }
  1311. }
  1312. while (! AllStarted);
  1313. return NO_ERROR;
  1314. }
  1315. VOID
  1316. ScHandleServiceFailure(
  1317. IN LPSERVICE_RECORD Service
  1318. )
  1319. /*++
  1320. Routine Description:
  1321. This function checks to see if the specified service failed to start.
  1322. If so, it clears the SERVICE_START_NOW flag, and determine if we
  1323. have to revert to last-known-good.
  1324. Arguments:
  1325. Service - Supplies a pointer to the service record to examine if
  1326. the service failed to start.
  1327. Return Value:
  1328. None.
  1329. --*/
  1330. {
  1331. if (Service->StartState == SC_START_FAIL) {
  1332. //
  1333. // Revert to last-known-good only if service is auto-start and
  1334. // fail to start due to reasons other than failure to logon.
  1335. //
  1336. if ((Service->ErrorControl == SERVICE_ERROR_SEVERE ||
  1337. Service->ErrorControl == SERVICE_ERROR_CRITICAL) &&
  1338. CURRENTSTART_FLAG_IS_SET(Service) &&
  1339. Service->StartError != ERROR_SERVICE_LOGON_FAILED) {
  1340. SC_LOG1(DEPEND,
  1341. "ScHandleServiceFailure: "
  1342. "About to call ScRevertToLastKnownGood for " FORMAT_LPWSTR
  1343. "\n", Service->ServiceName);
  1344. (void) ScRevertToLastKnownGood();
  1345. }
  1346. }
  1347. }
  1348. BOOL
  1349. ScDependenciesStarted(
  1350. IN LPSERVICE_RECORD Service,
  1351. OUT BOOL *IsBadDependencies,
  1352. OUT BOOL *AllStarted,
  1353. OUT BOOL *ExistsBlockedService
  1354. )
  1355. /*++
  1356. Routine Description:
  1357. This function checks to see if the dependencies of the specified
  1358. service are all started. If any of the dependencies has failed to
  1359. start, the specified service will be marked as failed to start.
  1360. If any of the dependencies is not marked as starting now (because
  1361. they are demand-start services), they are marked to be started now.
  1362. Arguments:
  1363. Service - Supplies a pointer to the service which we want to check
  1364. the start dependencies.
  1365. IsBadDependencies - Receives the value of TRUE if the service we
  1366. depend on belongs in a group that starts after the group we
  1367. are in. Otherwise, FALSE is returned.
  1368. AllStarted - Receives the value of FALSE if we have marked
  1369. a service as failed to be started because its dependent
  1370. didn't start. This means that our job of starting all services is
  1371. not done and we have to loop through an additional time to resolve
  1372. the state of any service that is dependent on it.
  1373. ExistsBlockedService - Receives the value of TRUE if a dependent
  1374. is not started or not failed to start. This indicates that
  1375. the specified service is still blocked from starting.
  1376. Return Value:
  1377. TRUE - if all dependencies have already been started successfully.
  1378. FALSE - if there exists one dependency that has not started or failed
  1379. to start.
  1380. --*/
  1381. {
  1382. BOOL AllDependenciesStarted = TRUE;
  1383. LPDEPEND_RECORD DependEntry;
  1384. LPSERVICE_RECORD DependService;
  1385. LPLOAD_ORDER_GROUP DependGroup;
  1386. DWORD GroupState;
  1387. for (DependEntry = Service->StartDepend;
  1388. DependEntry != NULL;
  1389. DependEntry = DependEntry->Next)
  1390. {
  1391. switch (DependEntry->DependType)
  1392. {
  1393. case TypeDependOnUnresolved:
  1394. //
  1395. // Error with service setup because it depends on a group or
  1396. // service which does not exists
  1397. //
  1398. SC_LOG2(ERROR, FORMAT_LPWSTR " depends on non-existing " FORMAT_LPWSTR
  1399. "\n", Service->DisplayName,
  1400. DependEntry->DependUnresolved->Name);
  1401. if (Service->ErrorControl != SERVICE_ERROR_IGNORE)
  1402. {
  1403. ScLogEvent(
  1404. NEVENT_SERVICE_START_FAILED_NONE,
  1405. Service->DisplayName,
  1406. DependEntry->DependUnresolved->Name
  1407. );
  1408. }
  1409. ScCleanupStartFailure(Service, ERROR_SERVICE_DEPENDENCY_DELETED);
  1410. *AllStarted = FALSE;
  1411. return FALSE;
  1412. case TypeDependOnService:
  1413. //
  1414. // Depend on a service
  1415. //
  1416. DependService = DependEntry->DependService;
  1417. //
  1418. // If dependency service already failed to start, the current service
  1419. // is set as failed to start.
  1420. //
  1421. if (DependService->StartState == SC_START_FAIL)
  1422. {
  1423. DWORD dwError;
  1424. SC_LOG3(ERROR, FORMAT_LPWSTR " depends on " FORMAT_LPWSTR
  1425. " which failed to start because " FORMAT_DWORD "\n",
  1426. Service->DisplayName, DependService->DisplayName,
  1427. (DependService->StartError != NO_ERROR) ?
  1428. DependService->StartError :
  1429. DependService->ServiceStatus.dwWin32ExitCode);
  1430. if (Service->ErrorControl != SERVICE_ERROR_IGNORE
  1431. &&
  1432. DependService->StartError != ERROR_NOT_SAFEBOOT_SERVICE
  1433. &&
  1434. !SetupInProgress(NULL, NULL))
  1435. {
  1436. ScLogEvent(
  1437. NEVENT_SERVICE_START_FAILED_II,
  1438. Service->DisplayName,
  1439. DependService->DisplayName,
  1440. DependService->StartError != NO_ERROR ?
  1441. DependService->StartError : DependService->ServiceStatus.dwWin32ExitCode
  1442. );
  1443. }
  1444. dwError = ERROR_NOT_SAFEBOOT_SERVICE;
  1445. //
  1446. // Walk through the list of service dependencies and pick
  1447. // the more appropriate error code. Do this to avoid
  1448. // reporting ERROR_SERVICE_DEPENDENCY_FAIL on auto-start
  1449. // and ERROR_NOT_SAFEBOOT_SERVICE on demand-start for the
  1450. // same service.
  1451. //
  1452. for (DependEntry = Service->StartDepend;
  1453. DependEntry != NULL;
  1454. DependEntry = DependEntry->Next)
  1455. {
  1456. if (DependEntry->DependType == TypeDependOnService
  1457. &&
  1458. DependEntry->DependService->StartError
  1459. != ERROR_NOT_SAFEBOOT_SERVICE)
  1460. {
  1461. //
  1462. // There was a "real" start failure
  1463. //
  1464. dwError = ERROR_SERVICE_DEPENDENCY_FAIL;
  1465. break;
  1466. }
  1467. }
  1468. ScCleanupStartFailure(Service, dwError);
  1469. *AllStarted = FALSE;
  1470. return FALSE;
  1471. }
  1472. if (DependService->StartState == SC_NEVER_STARTED)
  1473. {
  1474. *IsBadDependencies = IsDependOnLaterGroup(
  1475. Service->MemberOfGroup,
  1476. DependService->MemberOfGroup,
  1477. TypeDependOnService
  1478. );
  1479. if (*IsBadDependencies) {
  1480. //
  1481. // Circular dependency!
  1482. //
  1483. SC_LOG1(ERROR, "Circular dependency! " FORMAT_LPWSTR
  1484. " depends on service in a group which starts later\n",
  1485. Service->DisplayName);
  1486. if (Service->ErrorControl != SERVICE_ERROR_IGNORE)
  1487. {
  1488. ScLogEvent(
  1489. NEVENT_DEPEND_ON_LATER_SERVICE,
  1490. Service->DisplayName);
  1491. }
  1492. ScCleanupStartFailure(Service, ERROR_CIRCULAR_DEPENDENCY);
  1493. return FALSE;
  1494. }
  1495. //
  1496. // No circular dependency. Mark the dependency service
  1497. // as START_NOW.
  1498. //
  1499. DependService->StartState = SC_START_NOW;
  1500. DependService->StartError = NO_ERROR;
  1501. *AllStarted = FALSE;
  1502. }
  1503. //
  1504. // Get the current state of the dependency service
  1505. //
  1506. if (DependService->StartState != SC_START_SUCCESS)
  1507. {
  1508. AllDependenciesStarted = FALSE;
  1509. if (DependService->StartState != SC_START_FAIL)
  1510. {
  1511. //
  1512. // The current service is still blocked.
  1513. //
  1514. *ExistsBlockedService = TRUE;
  1515. }
  1516. }
  1517. break;
  1518. case TypeDependOnGroup:
  1519. //
  1520. // Depend on a group
  1521. //
  1522. DependGroup = DependEntry->DependGroup;
  1523. GroupState = ScGetCurrentGroupState(DependGroup);
  1524. switch (GroupState)
  1525. {
  1526. case GROUP_START_FAIL:
  1527. SC_LOG2(ERROR, FORMAT_LPWSTR " depends on failed group "
  1528. FORMAT_LPWSTR "\n", Service->DisplayName,
  1529. DependGroup->GroupName);
  1530. if (Service->ErrorControl != SERVICE_ERROR_IGNORE
  1531. &&
  1532. !g_SafeBootEnabled
  1533. &&
  1534. !SetupInProgress(NULL, NULL))
  1535. {
  1536. ScLogEvent(
  1537. NEVENT_SERVICE_START_FAILED_GROUP,
  1538. Service->DisplayName,
  1539. DependGroup->GroupName
  1540. );
  1541. }
  1542. ScCleanupStartFailure(Service, ERROR_SERVICE_DEPENDENCY_FAIL);
  1543. *AllStarted = FALSE;
  1544. return FALSE;
  1545. case GROUP_NOT_STARTED:
  1546. *IsBadDependencies = IsDependOnLaterGroup(
  1547. Service->MemberOfGroup,
  1548. DependGroup,
  1549. TypeDependOnGroup
  1550. );
  1551. if (*IsBadDependencies) {
  1552. //
  1553. // Circular dependency!
  1554. //
  1555. SC_LOG1(ERROR, "Circular dependency! " FORMAT_LPWSTR
  1556. " depends on a group which starts later\n",
  1557. Service->DisplayName);
  1558. if (Service->ErrorControl != SERVICE_ERROR_IGNORE)
  1559. {
  1560. ScLogEvent(
  1561. NEVENT_DEPEND_ON_LATER_GROUP,
  1562. Service->DisplayName
  1563. );
  1564. }
  1565. ScCleanupStartFailure(Service, ERROR_CIRCULAR_DEPENDENCY);
  1566. return FALSE;
  1567. }
  1568. //
  1569. // No circular dependency. Mark the services in the
  1570. // dependency group to START_NOW.
  1571. //
  1572. {
  1573. FOR_SERVICES_THAT(Svc,
  1574. BELONGS_TO(Svc, DependGroup) &&
  1575. Svc->StartState == SC_NEVER_STARTED)
  1576. {
  1577. Svc->StartState = SC_START_NOW;
  1578. Svc->StartError = NO_ERROR;
  1579. }
  1580. }
  1581. AllDependenciesStarted = FALSE;
  1582. *ExistsBlockedService = TRUE;
  1583. break;
  1584. default:
  1585. //
  1586. // Otherwise group must be started. Nothing to do.
  1587. //
  1588. SC_ASSERT(GroupState == GROUP_ONE_STARTED);
  1589. break;
  1590. }
  1591. break;
  1592. }
  1593. }
  1594. return AllDependenciesStarted;
  1595. }
  1596. BOOL
  1597. IsDependOnLaterGroup(
  1598. IN LPLOAD_ORDER_GROUP ServiceGroup,
  1599. IN LPLOAD_ORDER_GROUP DependOnGroup,
  1600. IN DEPEND_TYPE DependType
  1601. )
  1602. {
  1603. LPLOAD_ORDER_GROUP Group;
  1604. switch (DependType)
  1605. {
  1606. case TypeDependOnService:
  1607. if (ServiceGroup == DependOnGroup) {
  1608. //
  1609. // It is OK for a service to depend on another service
  1610. // in the same group.
  1611. //
  1612. return FALSE;
  1613. }
  1614. break;
  1615. case TypeDependOnGroup:
  1616. if (IS_SUBGROUP(ServiceGroup, DependOnGroup)) {
  1617. //
  1618. // It is circular dependency if a service depends on the
  1619. // group it itself belongs to
  1620. //
  1621. return TRUE;
  1622. }
  1623. break;
  1624. default:
  1625. SC_LOG(ERROR, "IsDependOnLaterGroup: got invalid DependType %lu\n",
  1626. DependType);
  1627. SC_ASSERT(FALSE);
  1628. return FALSE;
  1629. }
  1630. if (ServiceGroup == NULL ||
  1631. ServiceGroup->RefCount != MAXULONG ||
  1632. DependOnGroup == NULL ||
  1633. DependOnGroup->RefCount != MAXULONG) {
  1634. //
  1635. // Service we are starting belongs to a standalone group,
  1636. // or service or group we depend on is standalone.
  1637. //
  1638. return FALSE;
  1639. }
  1640. //
  1641. // Both the service's group and the depended on group are in the
  1642. // load order group list.
  1643. // The depended on group must not occur after the service's group.
  1644. // TDI GROUP SPECIAL: Also, if the depended on group is the TDI
  1645. // group, then there is an implicit dependency on the PNP_TDI group,
  1646. // so that must not occur after the service's group either.
  1647. //
  1648. for (Group = ServiceGroup->Next;
  1649. Group != NULL;
  1650. Group = Group->Next)
  1651. {
  1652. if (IS_SUBGROUP(Group, DependOnGroup)) {
  1653. return TRUE;
  1654. }
  1655. }
  1656. return FALSE;
  1657. }
  1658. VOID
  1659. ScCleanupStartFailure(
  1660. LPSERVICE_RECORD Service,
  1661. DWORD StartError
  1662. )
  1663. {
  1664. Service->StartState = SC_START_FAIL;
  1665. Service->StartError = StartError;
  1666. //
  1667. // Clear ERROR_SERVICE_NEVER_STARTED in the Win32ExitCode field if
  1668. // service failed to start.
  1669. //
  1670. CServiceRecordExclusiveLock RLock;
  1671. if (Service->ServiceStatus.dwWin32ExitCode ==
  1672. ERROR_SERVICE_NEVER_STARTED) {
  1673. Service->ServiceStatus.dwWin32ExitCode = StartError;
  1674. }
  1675. //
  1676. // For popup after user has logged on to indicate that some
  1677. // service started at boot has failed.
  1678. //
  1679. if (Service->ErrorControl != SERVICE_ERROR_IGNORE) {
  1680. ScPopupStartFail = TRUE;
  1681. }
  1682. }
  1683. VOID
  1684. ScLookForHungServices(
  1685. VOID
  1686. )
  1687. /*++
  1688. Routine Description:
  1689. This function loops through all services and queries the status
  1690. of each service that is start pending. It waits for the service to
  1691. show signs of progress in starting by waiting for the wait-hint
  1692. amount of time (in millisecs), and if the service is still start
  1693. pending and checkpoint has not incremented, the service's exitcode
  1694. is set to ERROR_SERVICE_START_HANG.
  1695. Arguments:
  1696. None.
  1697. Return Value:
  1698. None.
  1699. --*/
  1700. {
  1701. DWORD status;
  1702. SERVICE_STATUS CurrentServiceStatus;
  1703. STATUS_UNION ServiceStatus;
  1704. DWORD OldCheckPoint;
  1705. FOR_SERVICES_THAT(Service, Service->StartState == SC_START_PENDING)
  1706. {
  1707. ServiceStatus.Regular = &CurrentServiceStatus;
  1708. status = ScQueryServiceStatus(
  1709. Service,
  1710. ServiceStatus,
  1711. FALSE
  1712. );
  1713. if ((status == NO_ERROR) &&
  1714. (CurrentServiceStatus.dwCurrentState == SERVICE_START_PENDING)) {
  1715. #define SC_POLL_FACTOR 10
  1716. #define SC_MAX_SLEEP_TIME 10000
  1717. DWORD SleepTime = 1;
  1718. DWORD i;
  1719. DWORD NumIterations;
  1720. OldCheckPoint = CurrentServiceStatus.dwCheckPoint;
  1721. //
  1722. // Set up for the loop where we will poll the service status.
  1723. // The maximum sleep time during this polling operation will
  1724. // be 10 seconds.
  1725. //
  1726. //
  1727. // If the wait hint is greater than 100 seconds, then
  1728. // we want to modify the number of iterations through the
  1729. // loop so that we only sleep for the MAX_SLEEP_TIME.
  1730. //
  1731. // If the wait hint is less than that, then we change the
  1732. // sleep time to be less than 10 seconds, so that we go
  1733. // through the loop a max of 10 times.
  1734. //
  1735. if (CurrentServiceStatus.dwWaitHint > 100000) {
  1736. NumIterations = CurrentServiceStatus.dwWaitHint / SC_MAX_SLEEP_TIME;
  1737. SleepTime = SC_MAX_SLEEP_TIME;
  1738. }
  1739. else {
  1740. NumIterations = SC_POLL_FACTOR;
  1741. if (CurrentServiceStatus.dwWaitHint > SC_POLL_FACTOR) {
  1742. SleepTime = CurrentServiceStatus.dwWaitHint / SC_POLL_FACTOR;
  1743. }
  1744. }
  1745. for (i = 0; i < NumIterations; i++) {
  1746. //
  1747. // Wait a while for the checkpoint to increment, or
  1748. // service to be out of start-pending state.
  1749. //
  1750. Sleep(SleepTime);
  1751. status = ScQueryServiceStatus(
  1752. Service,
  1753. ServiceStatus,
  1754. FALSE
  1755. );
  1756. if (status == NO_ERROR) {
  1757. if (CurrentServiceStatus.dwCurrentState != SERVICE_START_PENDING ||
  1758. (CurrentServiceStatus.dwCurrentState == SERVICE_START_PENDING &&
  1759. OldCheckPoint < CurrentServiceStatus.dwCheckPoint)) {
  1760. goto NextService;
  1761. }
  1762. }
  1763. SC_LOG2(DEPEND, " Wait %ld on %ws for response\n", i + 1,
  1764. Service->ServiceName);
  1765. }
  1766. if ((status == NO_ERROR) &&
  1767. (CurrentServiceStatus.dwCurrentState == SERVICE_START_PENDING) &&
  1768. (OldCheckPoint == CurrentServiceStatus.dwCheckPoint)) {
  1769. SC_LOG2(ERROR, "%ws hung on starting (wait hint %lu ms)\n",
  1770. Service->ServiceName,
  1771. CurrentServiceStatus.dwWaitHint);
  1772. if (Service->ErrorControl != SERVICE_ERROR_IGNORE)
  1773. {
  1774. ScLogEvent(
  1775. NEVENT_SERVICE_START_HUNG,
  1776. Service->DisplayName
  1777. );
  1778. }
  1779. ScCleanupStartFailure(Service, ERROR_SERVICE_START_HANG);
  1780. }
  1781. }
  1782. if (status != NO_ERROR) {
  1783. SC_LOG2(ERROR, "ScLookForHungService: ScQueryServiceStatus "
  1784. FORMAT_LPWSTR " failed " FORMAT_DWORD "\n",
  1785. Service->ServiceName, status);
  1786. Service->StartState = SC_START_FAIL;
  1787. Service->StartError = ERROR_GEN_FAILURE;
  1788. }
  1789. NextService:
  1790. ;
  1791. }
  1792. }
  1793. VOID
  1794. ScHandleBadDependencies(
  1795. VOID
  1796. )
  1797. /*++
  1798. Routine Description:
  1799. This function is called when a circular dependency is detected.
  1800. Arguments:
  1801. None.
  1802. Return Value:
  1803. None.
  1804. --*/
  1805. {
  1806. FOR_SERVICES_THAT(Service,
  1807. (Service->StartState == SC_START_NOW) &&
  1808. (Service->ErrorControl == SERVICE_ERROR_SEVERE ||
  1809. Service->ErrorControl == SERVICE_ERROR_CRITICAL) )
  1810. {
  1811. SC_LOG1(ERROR, "ScHandleBadDependencies: "
  1812. "About to call ScRevertToLastKnownGood for "
  1813. FORMAT_LPWSTR "\n", Service->DisplayName);
  1814. ScLogEvent(
  1815. NEVENT_SEVERE_SERVICE_FAILED,
  1816. Service->DisplayName
  1817. );
  1818. ScRevertToLastKnownGood();
  1819. }
  1820. }
  1821. BOOL
  1822. ScServiceToStartDependOn(
  1823. LPSERVICE_RECORD ServiceToStart OPTIONAL,
  1824. LPSERVICE_RECORD StartPendingService
  1825. )
  1826. /*++
  1827. Routine Description:
  1828. This function is called by ScStartMarkedServices (BIG LOOP) to determine
  1829. if we have to wait for a pending service to complete.
  1830. If ServiceToStart == NULL, we are auto-starting service and we always
  1831. want to wait.
  1832. If ServiceToStart is not NULL, we are demand starting a service.
  1833. We have to wait if ServiceToStart depends on the StartPendingService.
  1834. Arguments:
  1835. ServiceToStart - Supplies the service record pointer of the service
  1836. being demand started.
  1837. StartPendingService - Supplies the service record pointer of the
  1838. service that is currently start pending.
  1839. Return Value:
  1840. TRUE - If ServiceToStart depends on StartPendingService or
  1841. ServiceToStart == NULL.
  1842. FALSE - Otherwise.
  1843. --*/
  1844. {
  1845. if (! ARGUMENT_PRESENT(ServiceToStart)) {
  1846. return TRUE;
  1847. }
  1848. if (ServiceToStart->StartState == SC_START_FAIL) {
  1849. return FALSE;
  1850. }
  1851. if (CURRENTSTART_FLAG_IS_SET(StartPendingService)) {
  1852. SC_LOG2(DEPEND_DUMP, "Service %ws directly/indirectly depends on pending service %ws\n",
  1853. ServiceToStart->ServiceName,
  1854. StartPendingService->ServiceName);
  1855. return TRUE;
  1856. }
  1857. SC_LOG(DEPEND_DUMP, "ScServiceToStartDependOn: Won't wait for pending "
  1858. FORMAT_LPWSTR "\n", StartPendingService->ServiceName);
  1859. return FALSE;
  1860. }
  1861. VOID
  1862. ScNotifyChangeState(
  1863. VOID
  1864. )
  1865. /*++
  1866. Routine Description:
  1867. This function is called by RSetServiceStatus which a service state
  1868. changes from start-pending to running or stopped. This will
  1869. notify the thread processing start dependencies that we can
  1870. proceed with starting up services that depend on the one that
  1871. called RSetServiceStatus.
  1872. Arguments:
  1873. None.
  1874. Return Value:
  1875. None.
  1876. --*/
  1877. {
  1878. if (! SetEvent(ScServiceChangeStateEvent)) {
  1879. SC_LOG1(ERROR, "ScNotifyChangeState: SetEvent error " FORMAT_DWORD "\n",
  1880. GetLastError());
  1881. SC_ASSERT(FALSE);
  1882. }
  1883. }
  1884. BOOL
  1885. ScDependentsStopped(
  1886. IN LPSERVICE_RECORD ServiceToStop
  1887. )
  1888. /*++
  1889. Routine Description:
  1890. This function checks to see if any service which depends on the
  1891. specified service is active. If so, it returns FALSE, otherwise
  1892. if no service depends on the specified service, or all services
  1893. which depend on the specified service is stopped, it returns
  1894. TRUE.
  1895. A service which is not in SERVICE_STOPPED is considered active.
  1896. Arguments:
  1897. ServiceToStop - Supplies a pointer to the service to see if other
  1898. active services depend on it.
  1899. Return Value:
  1900. TRUE - if all services which depend on ServiceToStop are stopped,
  1901. or there are no services which depend on ServiceToStop.
  1902. FALSE - if one or more of the services which depend on ServiceToStop
  1903. is active.
  1904. Note:
  1905. The database lock must be acquired with share access before calling
  1906. this routine.
  1907. --*/
  1908. {
  1909. SC_ASSERT(ScServiceRecordLock.Have());
  1910. LPDEPEND_RECORD StopDepend;
  1911. for (StopDepend = ServiceToStop->StopDepend;
  1912. StopDepend != NULL;
  1913. StopDepend = StopDepend->Next)
  1914. {
  1915. if (StopDepend->DependService->ServiceStatus.dwCurrentState
  1916. != SERVICE_STOPPED) {
  1917. SC_LOG1(DEPEND, FORMAT_LPWSTR " is still ACTIVE\n",
  1918. StopDepend->DependService->ServiceName);
  1919. return FALSE;
  1920. }
  1921. SC_LOG1(DEPEND, FORMAT_LPWSTR " is STOPPED\n",
  1922. StopDepend->DependService->ServiceName);
  1923. }
  1924. return TRUE;
  1925. }
  1926. VOID
  1927. ScEnumDependents(
  1928. IN LPSERVICE_RECORD ServiceRecord,
  1929. IN LPENUM_SERVICE_STATUS_WOW64 EnumBuffer,
  1930. IN DWORD RequestedState,
  1931. IN OUT LPDWORD EntriesRead,
  1932. IN OUT LPDWORD BytesNeeded,
  1933. IN OUT LPENUM_SERVICE_STATUS_WOW64 *EnumRecord,
  1934. IN OUT LPWSTR *EndOfVariableData,
  1935. IN OUT LPDWORD Status
  1936. )
  1937. /*++
  1938. Routine Description:
  1939. This function enumerates the stop depend list of the specified
  1940. service in the order which the dependents should be stopped.
  1941. Arguments:
  1942. ServiceRecord - Supplies a pointer to the service whose dependents
  1943. are to be enumerated.
  1944. EnumBuffer - Supplies a pointer to the first byte of the enum
  1945. buffer we are writing to. This is for duplicate entry checking.
  1946. RequestedState - Supplies one or the bitwise or of SERVICE_ACTIVE
  1947. and SERVICE_INACTIVE.
  1948. BytesNeeded - Supplies a pointer to a variable to receive the
  1949. running sum of bytes needed to enumerate all the entries.
  1950. EnumRecord - Supplies a pointer into the next location in the
  1951. output buffer to receive the next entry. The pointer is
  1952. updated on return.
  1953. EndOfVariableData - Supplies a pointer past the last available
  1954. byte in the output buffer so that variable length data
  1955. can be written from the end of the buffer. This pointer is
  1956. updated on return.
  1957. Status - Receives ERROR_MORE_DATA if dependent services does not
  1958. entirely fit in the output buffer. It should be initialized
  1959. to NO_ERROR this function is called.
  1960. Return Value:
  1961. None.
  1962. Note:
  1963. The database lock must be acquired with share access before calling
  1964. this routine.
  1965. --*/
  1966. {
  1967. SC_ASSERT(ScServiceRecordLock.Have());
  1968. LPDEPEND_RECORD StopDepend;
  1969. for (StopDepend = ServiceRecord->StopDepend;
  1970. StopDepend != NULL;
  1971. StopDepend = StopDepend->Next)
  1972. {
  1973. if (StopDepend->DependService->StopDepend != NULL) {
  1974. //
  1975. // Stop dependent also have other services that depends on
  1976. // it. Recursively call this routine to enumerate its
  1977. // dependents.
  1978. //
  1979. ScEnumDependents(
  1980. StopDepend->DependService,
  1981. EnumBuffer,
  1982. RequestedState,
  1983. EntriesRead,
  1984. BytesNeeded,
  1985. EnumRecord,
  1986. EndOfVariableData,
  1987. Status
  1988. );
  1989. }
  1990. if (
  1991. ((StopDepend->DependService->ServiceStatus.dwCurrentState
  1992. != SERVICE_STOPPED) &&
  1993. (RequestedState & SERVICE_ACTIVE))
  1994. ||
  1995. ((StopDepend->DependService->ServiceStatus.dwCurrentState
  1996. == SERVICE_STOPPED) &&
  1997. (RequestedState & SERVICE_INACTIVE))
  1998. ) {
  1999. SC_LOG1(DEPEND, "Enumerating dependent " FORMAT_LPWSTR "\n",
  2000. StopDepend->DependService->ServiceName);
  2001. if (! ScFoundDuplicateDependent(
  2002. StopDepend->DependService->ServiceName,
  2003. EnumBuffer,
  2004. *EnumRecord
  2005. )) {
  2006. *BytesNeeded += (sizeof(ENUM_SERVICE_STATUS_WOW64) +
  2007. (DWORD) WCSSIZE(StopDepend->DependService->ServiceName) +
  2008. (DWORD) WCSSIZE(StopDepend->DependService->DisplayName));
  2009. if (*Status == NO_ERROR) {
  2010. if (((DWORD_PTR) *EnumRecord + sizeof(ENUM_SERVICE_STATUS_WOW64)) >=
  2011. (DWORD_PTR) *EndOfVariableData) {
  2012. *Status = ERROR_MORE_DATA;
  2013. }
  2014. else {
  2015. //
  2016. // Write the entry into output buffer
  2017. //
  2018. memcpy(
  2019. (PVOID) &((*EnumRecord)->ServiceStatus),
  2020. (PVOID) &(StopDepend->DependService->ServiceStatus),
  2021. sizeof(SERVICE_STATUS)
  2022. );
  2023. //
  2024. // Copy the ServiceName string data
  2025. //
  2026. if (! ScCopyStringToBufferW(
  2027. StopDepend->DependService->ServiceName,
  2028. (DWORD) wcslen(StopDepend->DependService->ServiceName),
  2029. (LPWSTR) (*EnumRecord + 1),
  2030. EndOfVariableData,
  2031. (LPWSTR *) &((*EnumRecord)->dwServiceNameOffset),
  2032. (LPBYTE) EnumBuffer))
  2033. {
  2034. *Status = ERROR_MORE_DATA;
  2035. }
  2036. //
  2037. // Copy the DisplayName string data
  2038. //
  2039. if (! ScCopyStringToBufferW(
  2040. StopDepend->DependService->DisplayName,
  2041. (DWORD) wcslen(StopDepend->DependService->DisplayName),
  2042. (LPWSTR) ((*EnumRecord) + 1),
  2043. EndOfVariableData,
  2044. (LPWSTR *) &((*EnumRecord)->dwDisplayNameOffset),
  2045. (LPBYTE) EnumBuffer))
  2046. {
  2047. *Status = ERROR_MORE_DATA;
  2048. }
  2049. }
  2050. if (*Status == NO_ERROR) {
  2051. (*EnumRecord)++;
  2052. (*EntriesRead)++;
  2053. SC_LOG0(DEPEND, " Written into buffer successfully\n");
  2054. }
  2055. else {
  2056. SC_LOG0(DEPEND, " Failed to fit into buffer\n");
  2057. }
  2058. } // *Status is still NO_ERROR
  2059. } // non-duplicate entry
  2060. }
  2061. }
  2062. }
  2063. BOOL
  2064. ScFoundDuplicateDependent(
  2065. IN LPWSTR ServiceName,
  2066. IN LPENUM_SERVICE_STATUS_WOW64 EnumBuffer,
  2067. IN LPENUM_SERVICE_STATUS_WOW64 BufferEnd
  2068. )
  2069. /*++
  2070. Routine Description:
  2071. This function looks at service entries written to EnumBuffer for
  2072. any service names that matches the specified ServiceName.
  2073. Arguments:
  2074. ServiceName - Supplies the name of the service to look for.
  2075. EnumBuffer - Supplies a pointer to the buffer to look for matching
  2076. service name.
  2077. BufferEnd - Supplies a pointer to the end of buffer.
  2078. Return Value:
  2079. TRUE - if found a matching service name.
  2080. FALSE - no matching service name found.
  2081. --*/
  2082. {
  2083. LPENUM_SERVICE_STATUS_WOW64 EnumEntry;
  2084. for (EnumEntry = EnumBuffer;
  2085. EnumEntry < BufferEnd;
  2086. EnumEntry++)
  2087. {
  2088. if (_wcsicmp((LPWSTR) ((LPBYTE) EnumBuffer + EnumEntry->dwServiceNameOffset),
  2089. ServiceName) == 0)
  2090. {
  2091. return TRUE;
  2092. }
  2093. }
  2094. return FALSE;
  2095. }
  2096. #ifndef _CAIRO_
  2097. VOID
  2098. ScCheckNetLogonDepend(
  2099. LPSERVICE_RECORD ServiceRecord,
  2100. BOOL DemandStarting
  2101. )
  2102. /*++
  2103. Routine Description:
  2104. If the current service is running in a remote account or if we are
  2105. running on an Advanced Server (NtProductLanManNt), then this routine
  2106. makes a (soft) dependency on Netlogon. This dependency is not stored
  2107. in the registry.
  2108. Note that all services with account names that are UPNs will require
  2109. the dependency on Netlogon.
  2110. Arguments:
  2111. ServiceRecord - Pointer to the service record that is to be checked.
  2112. DemandStarting - boolean that indicates if we are demand starting or
  2113. auto starting.
  2114. Return Value:
  2115. none - If something fails within this function, we will just press on
  2116. since there isn't much we can do about it.
  2117. --*/
  2118. {
  2119. DWORD status;
  2120. HKEY ServiceNameKey;
  2121. LPWSTR DomainName;
  2122. BOOL bRemoteAccount=TRUE;
  2123. LPSERVICE_RECORD pNetLogonSR;
  2124. //
  2125. // Open the service name key.
  2126. //
  2127. status = ScOpenServiceConfigKey(
  2128. ServiceRecord->ServiceName,
  2129. KEY_READ,
  2130. FALSE, // Create if missing
  2131. &ServiceNameKey
  2132. );
  2133. if (status != NO_ERROR)
  2134. {
  2135. return;
  2136. }
  2137. //
  2138. // Read the account name from the registry.
  2139. //
  2140. status = ScReadStartName(
  2141. ServiceNameKey,
  2142. &DomainName
  2143. );
  2144. if (status != NO_ERROR || DomainName == NULL)
  2145. {
  2146. ScRegCloseKey(ServiceNameKey);
  2147. return;
  2148. }
  2149. ScRegCloseKey(ServiceNameKey);
  2150. if (_wcsicmp(DomainName, SC_LOCAL_SYSTEM_USER_NAME) == 0)
  2151. {
  2152. //
  2153. // LocalSystem account, we don't need netlogon.
  2154. //
  2155. SC_LOG1(TRACE,"ScCheckNetLogonDepend: %ws Service is LocalSystem!\n",
  2156. ServiceRecord->ServiceName);
  2157. LocalFree(DomainName);
  2158. return;
  2159. }
  2160. else if (wcsncmp(DomainName, L".\\", 2) == 0)
  2161. {
  2162. bRemoteAccount = FALSE;
  2163. SC_LOG1(TRACE,"ScCheckNetLogonDepend: %ws Service has a local domain name\n",
  2164. ServiceRecord->ServiceName);
  2165. }
  2166. else
  2167. {
  2168. //
  2169. // Check for local-only accounts (domain name of NT AUTHORITY). Note
  2170. // that the domain name may be quoted.
  2171. //
  2172. LPWSTR lpTempDomain = DomainName;
  2173. if (*lpTempDomain == '"')
  2174. {
  2175. lpTempDomain++;
  2176. }
  2177. if (_wcsnicmp(lpTempDomain,
  2178. SC_LOCAL_NTAUTH_NAME,
  2179. sizeof(SC_LOCAL_NTAUTH_NAME) / sizeof (WCHAR) - 1) == 0)
  2180. {
  2181. SC_LOG1(TRACE,
  2182. "ScCheckNetLogonDepend: %ws service is LocalService/NetworkService\n",
  2183. ServiceRecord->ServiceName);
  2184. LocalFree(DomainName);
  2185. return;
  2186. }
  2187. }
  2188. LocalFree(DomainName);
  2189. //
  2190. // We know if it runs in a remote account or not.
  2191. // Now we should check the product type. If it is an
  2192. // advanced server, or runs in an remote account, then
  2193. // we need to start NetLogon.
  2194. //
  2195. if ((ScGlobalProductType == NtProductLanManNt) || (bRemoteAccount)) {
  2196. //
  2197. // Get the service record for NetLogon.
  2198. //
  2199. status = ScGetNamedServiceRecord(L"NetLogon", &pNetLogonSR);
  2200. if (status != NO_ERROR) {
  2201. return;
  2202. }
  2203. //
  2204. // If it is already marked to start, then we don't
  2205. // have to do anything right now. If it isn't then
  2206. // we should SetServiceStartRequest and create a
  2207. // dependency.
  2208. //
  2209. if (CURRENTSTART_FLAG_IS_SET(pNetLogonSR)) {
  2210. //
  2211. // CurrentStartRequest of a dependency service is already
  2212. // set to TRUE. Just go on to next dependency.
  2213. //
  2214. SC_LOG2(DEPEND_DUMP, "DependService " FORMAT_LPWSTR
  2215. " CSR=TRUE already, USECOUNT=%lu\n",
  2216. pNetLogonSR->ServiceName,
  2217. pNetLogonSR->UseCount);
  2218. }
  2219. else {
  2220. LPDEPEND_RECORD pDependRecord;
  2221. SC_LOG(WHY, " Will start NetLogon because %ws runs in a remote account, or this is a server\n",
  2222. ServiceRecord->ServiceName);
  2223. ScSetServiceStartRequest(pNetLogonSR,DemandStarting);
  2224. //
  2225. // Add the dependency to the service record and mark it as
  2226. // temporary.
  2227. //
  2228. status = ScCreateDependRecord(TRUE,ServiceRecord,&pDependRecord);
  2229. if (status != NO_ERROR) {
  2230. return;
  2231. }
  2232. pDependRecord->DependType = TypeDependOnService;
  2233. pDependRecord->DependService = pNetLogonSR;
  2234. }
  2235. }
  2236. return;
  2237. }
  2238. #endif // _CAIRO_
  2239. BOOL
  2240. ScInHardwareProfile(
  2241. IN LPCWSTR ServiceName,
  2242. IN ULONG GetDeviceListFlags
  2243. )
  2244. /*++
  2245. Routine Description:
  2246. This function checks whether a specified service is enabled in the
  2247. current hardware profile.
  2248. Arguments:
  2249. Service - Specifies the service of interest.
  2250. GetDeviceListFlags - Specifies any special flags to be passed to
  2251. PNP_GetDeviceList. The CM_GETIDLIST_DONOTGENERATE
  2252. flag indicates that a legacy device instance should
  2253. not be generated for the service.
  2254. Return Value:
  2255. TRUE - if the service is enabled in the current hardware profile, or
  2256. if this cannot be determined.
  2257. FALSE - if the service is disabled in the current hardware profile.
  2258. --*/
  2259. {
  2260. CONFIGRET Status;
  2261. BOOL RetStatus;
  2262. WCHAR Buffer[200]; // default buffer on stack
  2263. WCHAR * pBuffer = Buffer;
  2264. ULONG cchLen;
  2265. LPCWSTR pDeviceID;
  2266. //
  2267. // Allocate a buffer for the list of device instances associated with
  2268. // this service
  2269. //
  2270. Status = PNP_GetDeviceListSize(
  2271. NULL, // hBinding
  2272. ServiceName, // pszFilter
  2273. &cchLen, // list length in wchars
  2274. CM_GETIDLIST_FILTER_SERVICE); // filter is a service name
  2275. if (Status != CR_SUCCESS)
  2276. {
  2277. SC_LOG2(WARNING, "PNP_GetDeviceListSize failed %#lx for service %ws\n",
  2278. Status, ServiceName);
  2279. return TRUE;
  2280. }
  2281. if (cchLen > LENGTH(Buffer))
  2282. {
  2283. SC_LOG2(DEPEND, "PNP_GetDeviceListSize wants a %lu-character buffer for service %ws\n",
  2284. cchLen, ServiceName);
  2285. pBuffer = (WCHAR *) LocalAlloc(0, cchLen * sizeof(WCHAR));
  2286. if (pBuffer == NULL)
  2287. {
  2288. SC_LOG(ERROR, "Couldn't allocate buffer for device list, error %lu\n",
  2289. GetLastError());
  2290. return TRUE;
  2291. }
  2292. }
  2293. else
  2294. {
  2295. cchLen = LENGTH(Buffer);
  2296. }
  2297. //
  2298. // Initialize parameters for PNP_GetDeviceList, the same way as is
  2299. // normally done in the client side of the API
  2300. //
  2301. pBuffer[0] = L'\0';
  2302. //
  2303. // Get the list of device instances that are associated with this service
  2304. //
  2305. // (For legacy services, the PNP manager makes up an artificial device
  2306. // instance; but for PNP-aware services, we could get an empty device list.)
  2307. //
  2308. Status = PNP_GetDeviceList(
  2309. NULL, // binding handle
  2310. ServiceName, // pszFilter
  2311. pBuffer, // buffer for device list
  2312. &cchLen, // buffer length in wchars
  2313. CM_GETIDLIST_FILTER_SERVICE | // filter is a service name
  2314. GetDeviceListFlags // OR with passed in flag
  2315. );
  2316. if (Status != CR_SUCCESS)
  2317. {
  2318. SC_LOG2(ERROR, "PNP_GetDeviceList failed %#lx for service %ws\n",
  2319. Status, ServiceName);
  2320. RetStatus = TRUE;
  2321. goto CleanExit;
  2322. }
  2323. //
  2324. // Get each device instance's config flags. The service is enabled in
  2325. // the current hardware profile if at least one of its devices is enabled.
  2326. //
  2327. for (pDeviceID = pBuffer;
  2328. pDeviceID[0] != L'\0';
  2329. pDeviceID += wcslen(pDeviceID) + 1)
  2330. {
  2331. ULONG ConfigFlags;
  2332. Status = PNP_HwProfFlags(
  2333. NULL, // binding handle
  2334. PNP_GET_HWPROFFLAGS, // action: get, not set
  2335. pDeviceID,
  2336. 0, // which profile: current one
  2337. &ConfigFlags,
  2338. NULL,
  2339. NULL,
  2340. 0,
  2341. 0 // flags, MBZ
  2342. );
  2343. if (Status == CR_SUCCESS)
  2344. {
  2345. if (!(ConfigFlags & (CSCONFIGFLAG_DISABLED |
  2346. CSCONFIGFLAG_DO_NOT_CREATE)))
  2347. {
  2348. //
  2349. // The device is enabled, so the service is enabled
  2350. //
  2351. RetStatus = TRUE;
  2352. goto CleanExit;
  2353. }
  2354. }
  2355. else
  2356. {
  2357. SC_LOG2(ERROR, "PNP_HwProfFlags failed %#lx for device %ws\n",
  2358. Status, pDeviceID);
  2359. }
  2360. }
  2361. RetStatus = FALSE;
  2362. SC_LOG(DEPEND, "The %ws service has no enabled device instances\n"
  2363. " in this hardware profile. Attempts to start\n"
  2364. " it will result in ERROR_SERVICE_DISABLED.\n",
  2365. ServiceName);
  2366. CleanExit:
  2367. if (pBuffer != Buffer)
  2368. {
  2369. LocalFree(pBuffer);
  2370. }
  2371. return RetStatus;
  2372. }