Информация об изменениях

Сообщение Правильная остановка долгозавершающейся службы от 30.03.2021 13:33

Изменено 01.04.2021 7:15 nekoriu

Правильная остановка долгозавершающейся службы
Добрый день.

Есть задача правильно завершить долгозавершающуюся рабочую службу. Служба моя.

Написал тестовую службу для отработки завершения, которая завершается 5 минут. Через 2 минуты после отправки команды stop приложение Cлужбы выдает ошибку, Служба не ответила на запрос своевременно.

Тестовая служба отправляет сообщение SERVICE_STOP_PENDING раз в секунду при завершении. Используется dwCheckPoint, dwWaitHint.

Windows 10.

Возможно ли без ошибок завершить тестовую службу в приложении Службы?

Часть ServiceMain.
while (1)
{
    WaitForSingleObject(ghSvcStopEvent, INFINITE);

    for (int i = 0; i < 300; i++)
    {
        ReportSvcStatus(SERVICE_STOP_PENDING, NO_ERROR, 2000);
        Sleep(1000);
    }

    ReportSvcStatus(SERVICE_STOPPED, NO_ERROR, 0);

    return;
}

Обработка сообщений от SCM.
VOID WINAPI SvcCtrlHandler( DWORD dwCtrl )
{
   switch(dwCtrl) 
   {  
      case SERVICE_CONTROL_STOP:
         ReportSvcStatus(SERVICE_STOP_PENDING, NO_ERROR, 0);

         // Signal the service to stop.
         SetEvent(ghSvcStopEvent);
         
         return;
 
      case SERVICE_CONTROL_INTERROGATE: 
         break; 
 
      default: 
         break;
   } 
}

Сообщение о статусе SCMу.
VOID ReportSvcStatus( DWORD dwCurrentState,
                      DWORD dwWin32ExitCode,
                      DWORD dwWaitHint)
{
    static DWORD dwCheckPoint = 1;

    gSvcStatus.dwCurrentState = dwCurrentState;
    gSvcStatus.dwWin32ExitCode = dwWin32ExitCode;
    gSvcStatus.dwWaitHint = dwWaitHint;

    if (dwCurrentState == SERVICE_START_PENDING)
        gSvcStatus.dwControlsAccepted = 0;
    else gSvcStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP;

    if ( (dwCurrentState == SERVICE_RUNNING) ||
           (dwCurrentState == SERVICE_STOPPED) )
        gSvcStatus.dwCheckPoint = 0;
    else gSvcStatus.dwCheckPoint = dwCheckPoint++;

    // Report the status of the service to the SCM.
    SetServiceStatus( gSvcStatusHandle, &gSvcStatus );
}
Правильная остановка долгозавершающейся службы
Добрый день.

Есть задача правильно завершить долгозавершающуюся рабочую службу. Служба моя.

Написал тестовую службу для отработки завершения, которая завершается 5 минут. Через 2 минуты после отправки команды stop приложение Cлужбы выдает ошибку, Служба не ответила на запрос своевременно.

Тестовая служба отправляет сообщение SERVICE_STOP_PENDING раз в секунду при завершении. Используется dwCheckPoint, dwWaitHint.

Windows 10.

Возможно ли без ошибок завершить тестовую службу в приложении Службы?

Часть ServiceMain.
while (1)
{
    WaitForSingleObject(ghSvcStopEvent, INFINITE);

    for (int i = 0; i < 300; i++)
    {
        ReportSvcStatus(SERVICE_STOP_PENDING, NO_ERROR, 2000);
        Sleep(1000);
    }

    ReportSvcStatus(SERVICE_STOPPED, NO_ERROR, 0);

    return;
}

Обработка сообщений от SCM.
VOID WINAPI SvcCtrlHandler( DWORD dwCtrl )
{
   switch(dwCtrl) 
   {  
      case SERVICE_CONTROL_STOP:
         ReportSvcStatus(SERVICE_STOP_PENDING, NO_ERROR, 0);

         // Signal the service to stop.
         SetEvent(ghSvcStopEvent);
         
         return;
 
      case SERVICE_CONTROL_INTERROGATE: 
     ReportSvcStatus(gSvcStatus.dwCurrentState, NO_ERROR, 0);
         break; 
 
      default: 
         break;
   } 
}

Сообщение о статусе SCMу.
VOID ReportSvcStatus( DWORD dwCurrentState,
                      DWORD dwWin32ExitCode,
                      DWORD dwWaitHint)
{
    static DWORD dwCheckPoint = 1;

    gSvcStatus.dwCurrentState = dwCurrentState;
    gSvcStatus.dwWin32ExitCode = dwWin32ExitCode;
    gSvcStatus.dwWaitHint = dwWaitHint;

    if (dwCurrentState == SERVICE_START_PENDING)
        gSvcStatus.dwControlsAccepted = 0;
    else gSvcStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP;

    if ( (dwCurrentState == SERVICE_RUNNING) ||
           (dwCurrentState == SERVICE_STOPPED) )
        gSvcStatus.dwCheckPoint = 0;
    else gSvcStatus.dwCheckPoint = dwCheckPoint++;

    // Report the status of the service to the SCM.
    SetServiceStatus( gSvcStatusHandle, &gSvcStatus );
}