-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathCServiceInstall.cpp
153 lines (134 loc) · 3.35 KB
/
CServiceInstall.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
#include "stdafx.h"
#include "resource.h"
#include "CServiceInstall.h"
CServiceInstall::CServiceInstall( LPCTSTR szServiceName, LPCTSTR szDisplay )
{
strcpy_s( m_service, sizeof( m_service ), szServiceName );
strcpy_s( m_display, sizeof( m_display ), szDisplay );
}
void CServiceInstall::Install(DWORD dwType, DWORD dwStart,
LPCTSTR lpDepends, LPCTSTR lpName,
LPCTSTR lpPassword, LPCTSTR lpDescription )
{
SC_HANDLE hSCM = NULL;
SC_HANDLE hService = NULL;
if( IsInstalled() == TRUE )
return;
hSCM = OpenSCManager( NULL,
NULL,
SC_MANAGER_CREATE_SERVICE );
if ( !hSCM ) {
ErrorPrinter( _TEXT("OpenSCManager") );
return;
}
TCHAR szFilePath[_MAX_PATH];
::GetModuleFileName(NULL, szFilePath, _MAX_PATH);
hService = ::CreateService(
hSCM,
m_service,
m_display,
SERVICE_ALL_ACCESS,
dwType,
dwStart,
SERVICE_ERROR_NORMAL,
szFilePath,
NULL,
NULL,
lpDepends,
lpName,
lpPassword );
if ( !hService ) {
ErrorPrinter( _TEXT("CreateService") );
::CloseServiceHandle( hSCM );
return;
}
if( lpDescription != NULL ) {
_tprintf(_TEXT("%s Created\n"), m_service);
HMODULE hmod = ::LoadLibrary("Advapi32.dll");
if(hmod != NULL) {
if(::GetProcAddress(hmod, "ChangeServiceConfig2A") != NULL) {
SERVICE_DESCRIPTION sd;
sd.lpDescription = _strdup( lpDescription );
::ChangeServiceConfig2(hService, SERVICE_CONFIG_DESCRIPTION, &sd);
free( sd.lpDescription );
}
::FreeLibrary(hmod);
}
}
if ( !::StartService( hService, 0, NULL ) ) {
ErrorPrinter( _TEXT("StartService") );
}
::CloseServiceHandle( hService );
::CloseServiceHandle( hSCM );
return;
}
void CServiceInstall::Remove( BOOL bForce )
{
SC_HANDLE hSCM = NULL;
SC_HANDLE hService = NULL;
BOOL bSuccess = FALSE;
hSCM = ::OpenSCManager( NULL,
NULL,
SC_MANAGER_CONNECT );
if ( !hSCM ) {
ErrorPrinter( _TEXT("OpenSCManager") );
return;
}
hService = ::OpenService(
hSCM,
m_service,
SERVICE_STOP | DELETE );
if ( !hService ) {
ErrorPrinter( _TEXT("OpenService") );
if (hService) CloseServiceHandle( hService );
if (hSCM) CloseServiceHandle( hSCM );
return;
}
//Force the service to stop
if( bForce == TRUE ) {
SERVICE_STATUS status;
::ControlService(hService, SERVICE_CONTROL_STOP, &status);
_tprintf(_TEXT("%s stopped\n"), m_service);
}
bSuccess = ::DeleteService( hService );
if (bSuccess) {
_tprintf(_TEXT("%s removed\n"), m_service);
} else {
ErrorPrinter( _TEXT("DeleteService") );
}
if (hService) CloseServiceHandle( hService );
if (hSCM) CloseServiceHandle( hSCM );
return;
}
BOOL CServiceInstall::IsInstalled( )
{
BOOL bRes = FALSE;
SC_HANDLE hSCM = ::OpenSCManager(NULL, NULL, SC_MANAGER_CONNECT);
if (hSCM != NULL)
{
SC_HANDLE hService = ::OpenService(hSCM, m_service, SERVICE_QUERY_CONFIG);
if (hService != NULL)
{
bRes = TRUE;
::CloseServiceHandle(hService);
}
::CloseServiceHandle(hSCM);
}
return bRes;
}
DWORD CServiceInstall::ErrorPrinter( const TCHAR* psz, DWORD dwErr )
{
LPVOID lpvMsgBuf;
if ( !FormatMessage( FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER,
0, dwErr,
MAKELANGID( LANG_NEUTRAL, SUBLANG_DEFAULT ),
(LPTSTR) &lpvMsgBuf, 0, 0 ) )
{
_tprintf( _TEXT("%s failed: Unknown error %x\n"), psz, dwErr );
}
else {
_tprintf( _TEXT("%s failed: %s\n"), psz, (LPTSTR)lpvMsgBuf );
}
LocalFree( lpvMsgBuf );
return dwErr;
}