Добрый день , коллеги, хочется найти простой пример создания виндусовского сервиса. Интернет облазил, нашел один пример, но после некоторых манипуляций и исправления ошибок одна остается и как с ней бороться не пойму, киньте рабочий код простого сервиса или поправьте меня, где я не прав в коде:
#include <Windows.h>
#include <tchar.h>
#include <string>
#include <cstdio>
#define SERVICE_NAME _T("My Sample Service")
SERVICE_STATUS g_ServiceStatus = { 0 };
SERVICE_STATUS_HANDLE g_StatusHandle = NULL;
HANDLE g_ServiceStopEvent = INVALID_HANDLE_VALUE;
VOID WINAPI ServiceMain(DWORD argc, LPTSTR* argv);
VOID WINAPI ServiceCtrlHandler(DWORD);
DWORD WINAPI ServiceWorkerThread(LPVOID lpParam);
VOID WINAPI ServiceMain(DWORD argc, LPTSTR* argv)
{
DWORD Status = E_FAIL;
// Register our service control handler with the SCM
g_StatusHandle = RegisterServiceCtrlHandler(SERVICE_NAME, ServiceCtrlHandler);
if (g_StatusHandle == NULL)
{
goto EXIT;
}
// Tell the service controller we are starting
ZeroMemory(&g_ServiceStatus, sizeof(g_ServiceStatus));
g_ServiceStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
g_ServiceStatus.dwControlsAccepted = 0;
g_ServiceStatus.dwCurrentState = SERVICE_START_PENDING;
g_ServiceStatus.dwWin32ExitCode = 0;
g_ServiceStatus.dwServiceSpecificExitCode = 0;
g_ServiceStatus.dwCheckPoint = 0;
if (SetServiceStatus(g_StatusHandle, &g_ServiceStatus) == FALSE)
{
OutputDebugString(_T(
"My Sample Service: ServiceMain: SetServiceStatus returned error"));
}
/*
* Perform tasks necessary to start the service here
*/
// Create a service stop event to wait on later
g_ServiceStopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (g_ServiceStopEvent == NULL)
{
// Error creating event
// Tell service controller we are stopped and exit
g_ServiceStatus.dwControlsAccepted = 0;
g_ServiceStatus.dwCurrentState = SERVICE_STOPPED;
g_ServiceStatus.dwWin32ExitCode = GetLastError();
g_ServiceStatus.dwCheckPoint = 1;
if (SetServiceStatus(g_StatusHandle, &g_ServiceStatus) == FALSE)
{
OutputDebugString(_T(
"My Sample Service: ServiceMain: SetServiceStatus returned error"));
}
goto EXIT;
}
// Tell the service controller we are started
g_ServiceStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP;
g_ServiceStatus.dwCurrentState = SERVICE_RUNNING;
g_ServiceStatus.dwWin32ExitCode = 0;
g_ServiceStatus.dwCheckPoint = 0;
if (SetServiceStatus(g_StatusHandle, &g_ServiceStatus) == FALSE)
{
OutputDebugString(_T(
"My Sample Service: ServiceMain: SetServiceStatus returned error"));
}
// Start a thread that will perform the main task of the service
HANDLE hThread = CreateThread(NULL, 0, ServiceWorkerThread, NULL, 0, NULL);
// Wait until our worker thread exits signaling that the service needs to stop
WaitForSingleObject(hThread, INFINITE);
/*
* Perform any cleanup tasks
*/
CloseHandle(g_ServiceStopEvent);
// Tell the service controller we are stopped
g_ServiceStatus.dwControlsAccepted = 0;
g_ServiceStatus.dwCurrentState = SERVICE_STOPPED;
g_ServiceStatus.dwWin32ExitCode = 0;
g_ServiceStatus.dwCheckPoint = 3;
if (SetServiceStatus(g_StatusHandle, &g_ServiceStatus) == FALSE)
{
OutputDebugString(_T(
"My Sample Service: ServiceMain: SetServiceStatus returned error"));
}
EXIT:
return;
}
int _tmain(int argc, TCHAR* argv[])
{
SERVICE_TABLE_ENTRY ServiceTable[] =
{
{(LPWSTR)SERVICE_NAME, (LPSERVICE_MAIN_FUNCTION)ServiceMain},
{NULL, NULL}
};
if (StartServiceCtrlDispatcher(ServiceTable) == FALSE)
{
return GetLastError();
}
return 0;
}
Ошибка с goto Error C2362 initialization of 'hThread' is skipped by 'goto EXIT'
Здравствуйте, CaptainFlint, Вы писали:
CF>Здравствуйте, arfaa, Вы писали:
A>>Ошибка с goto A>>Error C2362 initialization of 'hThread' is skipped by 'goto EXIT'
CF>Вынеси объявление hThread за пределы goto:
CF>
Здравствуйте, arfaa, Вы писали:
A>Здравствуйте, CaptainFlint, Вы писали:
CF>>Здравствуйте, arfaa, Вы писали:
A>>>Ошибка с goto A>>>Error C2362 initialization of 'hThread' is skipped by 'goto EXIT'
CF>>Вынеси объявление hThread за пределы goto:
CF>>
Здравствуйте, arfaa, Вы писали:
A>Error LNK2019 unresolved external symbol "void __cdecl ServiceCtrlHandler(unsigned long)" (?ServiceCtrlHandler@@YAXK@Z) referenced in function "void __cdecl ServiceMain(unsigned long,wchar_t * *)" (?ServiceMain@@YAXKPEAPEA_W@Z) ConsoleApplication3
A>Error LNK2019 unresolved external symbol "unsigned long __cdecl ServiceWorkerThread(void *)" (?ServiceWorkerThread@@YAKPEAX@Z) referenced in function "void __cdecl ServiceMain(unsigned long,wchar_t * *)" (?ServiceMain@@YAXKPEAPEA_W@Z)
Мне кажется, стоит сначала разобраться в основах программирования на языке C/C++, и лишь потом браться за написание сервисов.
Эти две функции объявлены, используются в коде, но тела самих этих функций нигде не определены. Либо не хватает кода в примере, либо не подключен дополнительный файл с исходником.
Здравствуйте, arfaa, Вы писали:
A>Добрый день , коллеги, хочется найти простой пример создания виндусовского сервиса. Интернет облазил, нашел один пример, но после некоторых манипуляций и исправления ошибок одна остается и как с ней бороться не пойму, киньте рабочий код простого сервиса или поправьте меня, где я не прав в коде:
Когда-то в начале нулевых я писал виндовые сервисы на Delphi. Вам то же самое рекомендую сделать, взяв скажем Lazarus. Оно будет проще и вам не придётся разбираться одновременно с адскими С++ и с winapi.
И я бы добавил в ваш код две новые функции — регистрация собственно сервиса и удаление регистрации. Чтобы он был самодостаточным.
Здравствуйте, arfaa, Вы писали:
A>Добрый день , коллеги, хочется найти простой пример создания виндусовского сервиса. Интернет облазил, нашел один пример, но после некоторых манипуляций и исправления ошибок одна остается и как с ней бороться не пойму, киньте рабочий код простого сервиса или поправьте меня, где я не прав в коде:
Здравствуйте, Alexander_S_U, Вы писали:
A_S>Здравствуйте, arfaa, Вы писали:
A>>Добрый день , коллеги, хочется найти простой пример создания виндусовского сервиса. Интернет облазил, нашел один пример, но после некоторых манипуляций и исправления ошибок одна остается и как с ней бороться не пойму, киньте рабочий код простого сервиса или поправьте меня, где я не прав в коде:
A_S>Посмотрите во эту статью Пишем сервис. Что может быть проще!
Здравствуйте, Слава, Вы писали:
С>Здравствуйте, arfaa, Вы писали:
A>>Добрый день , коллеги, хочется найти простой пример создания виндусовского сервиса. Интернет облазил, нашел один пример, но после некоторых манипуляций и исправления ошибок одна остается и как с ней бороться не пойму, киньте рабочий код простого сервиса или поправьте меня, где я не прав в коде:
С>Когда-то в начале нулевых я писал виндовые сервисы на Delphi. Вам то же самое рекомендую сделать, взяв скажем Lazarus. Оно будет проще и вам не придётся разбираться одновременно с адскими С++ и с winapi.
С>И я бы добавил в ваш код две новые функции — регистрация собственно сервиса и удаление регистрации. Чтобы он был самодостаточным.
Спасибо, что не оставили тоже без ответа!
Здравствуйте, arfaa, Вы писали:
A>Добрый день , коллеги, хочется найти простой пример создания виндусовского сервиса. Интернет облазил, нашел один пример, но ...
Я бы порекомендовал все же официальную документацию:
A> // Start a thread that will perform the main task of the service
A> HANDLE hThread = CreateThread(NULL, 0, ServiceWorkerThread, NULL, 0, NULL);
A> // Wait until our worker thread exits signaling that the service needs to stop
A> WaitForSingleObject(hThread, INFINITE);
A>
А зачем создавать отдельную нитку, и сразу же уходить в бесконечное ожидание ее завершения? Почему бы все то, что делает эта нитка, не делать в контексте основной нитки?
A>Ошибка с goto A>Error C2362 initialization of 'hThread' is skipped by 'goto EXIT'
goto не должен обходить инициализацию переменных. Иначе если исполнение пойдет по пути "goto", то в том месте, куда goto перейдет, переменная будет неинициализированная.
Здравствуйте, Pzz, Вы писали:
Pzz>Здравствуйте, arfaa, Вы писали:
A>>
A>> // Start a thread that will perform the main task of the service
A>> HANDLE hThread = CreateThread(NULL, 0, ServiceWorkerThread, NULL, 0, NULL);
A>> // Wait until our worker thread exits signaling that the service needs to stop
A>> WaitForSingleObject(hThread, INFINITE);
A>>
Pzz>А зачем создавать отдельную нитку, и сразу же уходить в бесконечное ожидание ее завершения? Почему бы все то, что делает эта нитка, не делать в контексте основной нитки?
A>>Ошибка с goto A>>Error C2362 initialization of 'hThread' is skipped by 'goto EXIT'
Pzz>goto не должен обходить инициализацию переменных. Иначе если исполнение пойдет по пути "goto", то в том месте, куда goto перейдет, переменная будет неинициализированная.
1. попробуйте тут получить LastError ( addLogMessage("Error: Can't start service"); )
2. попробуйте запустить ваш сервис сторонним приложением, тем же SCM от Рихтера или банально через консоль (sc start ), что будет писать.
Может, проблема в нем самом.
Здравствуйте, morgot, Вы писали:
M>Здравствуйте, arfaa, Вы писали:
M>1. попробуйте тут получить LastError ( addLogMessage("Error: Can't start service"); ) M>2. попробуйте запустить ваш сервис сторонним приложением, тем же SCM от Рихтера или банально через консоль (sc start ), что будет писать. M>Может, проблема в нем самом.
Реализовал сервис, который корректно работает, вот код:
#include <windows.h>
#include <tchar.h>
#include <strsafe.h>
#include <iostream>
#include <fstream>
#pragma comment(lib, "advapi32.lib")
//#define SVCNAME TEXT("SvcName")
SERVICE_STATUS gSvcStatus;
SERVICE_STATUS_HANDLE gSvcStatusHandle;
HANDLE ghSvcStopEvent = NULL;
VOID SvcInstall(void);
VOID WINAPI SvcCtrlHandler(DWORD);
VOID WINAPI SvcMain(DWORD, LPTSTR*);
VOID ReportSvcStatus(DWORD, DWORD, DWORD);
VOID SvcInit(DWORD, LPTSTR*);
VOID SvcReportEvent(LPTSTR);
using namespace std;
//
// Purpose:
// Entry point for the process
//
// Parameters:
// None
//
// Return value:
// None
//void __cdecl _tmain(int argc, TCHAR* argv[])
{
// If command-line parameter is "install", install the service.
// Otherwise, the service is probably being started by the SCM.if (lstrcmpi(argv[1], TEXT("install")) == 0)
{
//ofstream file;
// file.open("codebind.txt", std::fstream::app);
// file << "Please writr this text to a file.\n this text is written using C---\n";
// file.close();
SvcInstall();
return;
}
wchar_t SVCNAME[100];
swprintf(SVCNAME, 100, L"SimpleService3");
// TO_DO: Add any additional services for the process to this table.
SERVICE_TABLE_ENTRY DispatchTable[] =
{
{ SVCNAME, (LPSERVICE_MAIN_FUNCTION)SvcMain },
{ NULL, NULL }
};
// This call returns when the service has stopped.
// The process should simply terminate when the call returns.if (!StartServiceCtrlDispatcher(DispatchTable))
{
//SvcReportEvent(TEXT("StartServiceCtrlDispatcher"));
}
}
//
// Purpose:
// Installs a service in the SCM database
//
// Parameters:
// None
//
// Return value:
// None
//
VOID SvcInstall()
{
string str = "erere";
SC_HANDLE schSCManager;
SC_HANDLE schService;
TCHAR szPath[MAX_PATH];
if (!GetModuleFileName(NULL, szPath, MAX_PATH))
{
printf("Cannot install service (%d)\n", GetLastError());
return;
}
// Get a handle to the SCM database.
schSCManager = OpenSCManager(
NULL, // local computer
NULL, // ServicesActive database
SC_MANAGER_ALL_ACCESS); // full access rights if (NULL == schSCManager)
{
printf("OpenSCManager failed (%d)\n", GetLastError());
return;
}
// Create the servicewchar_t SVCNAME[100];
swprintf(SVCNAME, 100, L"SimpleService3");
schService = CreateService(
schSCManager, // SCM database
SVCNAME, // name of service
SVCNAME, // service name to display
SERVICE_ALL_ACCESS, // desired access
SERVICE_WIN32_OWN_PROCESS, // service type
SERVICE_DEMAND_START, // start type
SERVICE_ERROR_NORMAL, // error control type
szPath, // path to service's binary
NULL, // no load ordering group
NULL, // no tag identifier
NULL, // no dependencies
NULL, // LocalSystem account
NULL); // no password if (schService == NULL)
{
printf("CreateService failed (%d)\n", GetLastError());
CloseServiceHandle(schSCManager);
return;
}
else printf("Service installed successfully\n");
CloseServiceHandle(schService);
CloseServiceHandle(schSCManager);
}
//
// Purpose:
// Entry point for the service
//
// Parameters:
// dwArgc - Number of arguments in the lpszArgv array
// lpszArgv - Array of strings. The first string is the name of
// the service and subsequent strings are passed by the process
// that called the StartService function to start the service.
//
// Return value:
// None.
//
VOID WINAPI SvcMain(DWORD dwArgc, LPTSTR* lpszArgv)
{
// Register the handler function for the servicewchar_t SVCNAME[100];
swprintf(SVCNAME, 100, L"SimpleService3");
gSvcStatusHandle = RegisterServiceCtrlHandler(
SVCNAME,
SvcCtrlHandler);
if (!gSvcStatusHandle)
{
//SvcReportEvent(TEXT("RegisterServiceCtrlHandler"));return;
}
// These SERVICE_STATUS members remain as set here
gSvcStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
gSvcStatus.dwServiceSpecificExitCode = 0;
// Report initial status to the SCM
ReportSvcStatus(SERVICE_START_PENDING, NO_ERROR, 3000);
// Perform service-specific initialization and work.
SvcInit(dwArgc, lpszArgv);
}
//
// Purpose:
// The service code
//
// Parameters:
// dwArgc - Number of arguments in the lpszArgv array
// lpszArgv - Array of strings. The first string is the name of
// the service and subsequent strings are passed by the process
// that called the StartService function to start the service.
//
// Return value:
// None
//
VOID SvcInit(DWORD dwArgc, LPTSTR* lpszArgv)
{
// TO_DO: Declare and set any required variables.
// Be sure to periodically call ReportSvcStatus() with
// SERVICE_START_PENDING. If initialization fails, call
// ReportSvcStatus with SERVICE_STOPPED.
// Create an event. The control handler function, SvcCtrlHandler,
// signals this event when it receives the stop control code.
ghSvcStopEvent = CreateEvent(
NULL, // default security attributes
TRUE, // manual reset event
FALSE, // not signaled
NULL); // no nameif (ghSvcStopEvent == NULL)
{
ReportSvcStatus(SERVICE_STOPPED, NO_ERROR, 0);
return;
}
// Report running status when initialization is complete.
ReportSvcStatus(SERVICE_RUNNING, NO_ERROR, 0);
// TO_DO: Perform work until service stops.while (1)
{
// Check whether to stop the service.
WaitForSingleObject(ghSvcStopEvent, INFINITE);
ReportSvcStatus(SERVICE_STOPPED, NO_ERROR, 0);
return;
}
}
//
// Purpose:
// Sets the current service status and reports it to the SCM.
//
// Parameters:
// dwCurrentState - The current state (see SERVICE_STATUS)
// dwWin32ExitCode - The system error code
// dwWaitHint - Estimated time for pending operation,
// in milliseconds
//
// Return value:
// None
//
VOID ReportSvcStatus(DWORD dwCurrentState,
DWORD dwWin32ExitCode,
DWORD dwWaitHint)
{
static DWORD dwCheckPoint = 1;
// Fill in the SERVICE_STATUS structure.
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);
}
//
// Purpose:
// Called by SCM whenever a control code is sent to the service
// using the ControlService function.
//
// Parameters:
// dwCtrl - control code
//
// Return value:
// None
//
VOID WINAPI SvcCtrlHandler(DWORD dwCtrl)
{
// Handle the requested control code. switch (dwCtrl)
{
case SERVICE_CONTROL_STOP:
ReportSvcStatus(SERVICE_STOP_PENDING, NO_ERROR, 0);
// Signal the service to stop.
SetEvent(ghSvcStopEvent);
ReportSvcStatus(gSvcStatus.dwCurrentState, NO_ERROR, 0);
return;
case SERVICE_CONTROL_INTERROGATE:
break;
default:
break;
}
}
//
// Purpose:
// Logs messages to the event log
//
// Parameters:
// szFunction - name of function that failed
//
// Return value:
// None
//
// Remarks:
// The service must have an entry in the Application event log.
//
VOID SvcReportEvent(LPTSTR szFunction)
{
HANDLE hEventSource;
LPCTSTR lpszStrings[2];
TCHAR Buffer[80];
wchar_t SVCNAME[100];
swprintf(SVCNAME, 100, L"SimpleService3");
hEventSource = RegisterEventSource(NULL, SVCNAME);
if (NULL != hEventSource)
{
StringCchPrintf(Buffer, 80, TEXT("%s failed with %d"), szFunction, GetLastError());
lpszStrings[0] = SVCNAME;
lpszStrings[1] = Buffer;
ReportEvent(hEventSource, // event log handle
EVENTLOG_ERROR_TYPE, // event type
0, // event category
0,//SVC_ERROR, // event identifier
NULL, // no security identifier
2, // size of lpszStrings array
0, // no binary data
lpszStrings, // array of strings
NULL); // no binary data
DeregisterEventSource(hEventSource);
}
}