-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathCservice.cpp
331 lines (270 loc) · 7.38 KB
/
Cservice.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
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
#include "stdafx.h"
#include "cservice.h"
CService::CService(LPCTSTR szName, LPCTSTR szDisplay, DWORD dwType) : m_dwType(dwType)
{
bDebug = false;
m_hServiceStatus = NULL;
m_dwRequestedControl = 0;
// Control Events
m_hWatcherThread = NULL;
m_dwState = 0;
m_dwControlsAccepted = 0;
m_dwCheckpoint = 0;
m_dwWaitHint = 0;
// Initialize event handles to NULL
for(int i = 0 ; i < NUMEVENTS ; i++)
m_hEvents[i] = NULL;
// Copy string names
strcpy_s( m_szName, sizeof( m_szName ), szName );
strcpy_s( m_szDisplay, sizeof( m_szDisplay ), szDisplay );
// Set up class critical section
InitializeCriticalSection(&m_cs);
}
CService::~CService()
{
DeleteCriticalSection(&m_cs);
}
void CService::PreInit()
{
// Initialize Events
for(int i = 0 ; i < NUMEVENTS ; i++)
{
m_hEvents[i] = CreateEvent(NULL, FALSE, FALSE, NULL);
if(!m_hEvents[i])
ErrorHandler(_T("CreateEvent"));
}
}
void CService::DeInit()
{
// Wait for the watcher thread to terminate
if(m_hWatcherThread)
{
// Wait a reasonable amount of time
DWORD rt = WaitForSingleObject(m_hWatcherThread, 10000);
CloseHandle(m_hWatcherThread);
}
// Uninitialize any resources created in Init()
for(int i = 0 ; i < NUMEVENTS ; i++)
{
if(m_hEvents[i])
CloseHandle(m_hEvents[i]);
}
}
void CService::ServiceMainMember(DWORD argc, LPTSTR* argv, LPHANDLER_FUNCTION pf, LPTHREAD_START_ROUTINE pfnWTP)
{
DWORD dwErr = 0;
#ifndef _DEBUG
__try
{
#endif
PreInit();
SetupHandlerInside(pf);
ParseArgs(argc, argv);
LaunchWatcherThread(pfnWTP);
Init();
Run();
#ifndef _DEBUG
}
__except(dwErr = GetExceptionCode(), EXCEPTION_EXECUTE_HANDLER)
{
if(m_hServiceStatus)
SetStatus(SERVICE_STOPPED, 0, 0, 0, dwErr, 0);
}
#endif
DeInit();
SetStatus(SERVICE_STOPPED, 0, 0, 0, dwErr, 0);
}
void CService::DebugMainMember(DWORD argc, LPTSTR* argv, LPTHREAD_START_ROUTINE pfnWTP)
{
DWORD dwErr = 0;
#ifndef _DEBUG
__try
{
#endif
PreInit();
ParseArgs(argc, argv);
LaunchWatcherThread(pfnWTP);
Init();
Run();
#ifndef _DEBUG
}
__except(dwErr = GetExceptionCode(), EXCEPTION_EXECUTE_HANDLER)
{
if(m_hServiceStatus)
SetStatus(SERVICE_STOPPED, 0, 0, 0, dwErr, 0);
}
#endif
DeInit();
SetStatus(SERVICE_STOPPED, 0, 0, 0, dwErr, 0);
}
// Register the control handler for the service
bool CService::SetupHandlerInside(LPHANDLER_FUNCTION lpHandlerProc)
{
m_hServiceStatus = RegisterServiceCtrlHandler(m_szName, lpHandlerProc);
if(!m_hServiceStatus)
{
ErrorHandler(_T("RegisterServiceCtrlHandler"));
return false;
}
SetStatus(SERVICE_START_PENDING, 1, 5000);
return true;
}
void CService::HandlerMember(DWORD dwControl)
{
// Keep an additional control request of the same type
// from coming in when you're already handling it
if(m_dwRequestedControl == dwControl)
return;
switch(dwControl)
{
case SERVICE_CONTROL_STOP:
m_dwRequestedControl = dwControl;
// Notify the service to stop...
SetEvent(m_hEvents[STOP]);
break;
case SERVICE_CONTROL_PAUSE:
m_dwRequestedControl = dwControl;
// Notify the service to pause...
SetEvent(m_hEvents[PAUSE]);
break;
case SERVICE_CONTROL_CONTINUE:
if(GetStatus() != SERVICE_RUNNING)
{
m_dwRequestedControl = dwControl;
// Notify the service to continue...
SetEvent(m_hEvents[CONTINUE]);
}
break;
case SERVICE_CONTROL_SHUTDOWN:
m_dwRequestedControl = dwControl;
SetEvent(m_hEvents[SHUTDOWN]);
break;
case SERVICE_CONTROL_INTERROGATE:
// Return current status on interrogation
SetStatus(GetStatus());
break;
default: // User Defined
m_dwRequestedControl = dwControl;
HandleUserDefined(dwControl);
}
}
void CService::LaunchWatcherThread(LPTHREAD_START_ROUTINE pfnWTP)
{
DWORD tid = 0;
m_hWatcherThread = (HANDLE)_beginthreadex(0, 0, (unsigned (WINAPI*)(void*))pfnWTP, 0, 0, (unsigned int*)&tid);
if(!m_hWatcherThread)
ErrorHandler(_T("_beginthreadex"));
}
DWORD CService::WatcherThreadMemberProc()
{
DWORD dwWait = 0;
bool bControlWait = true;
// Wait for any events to signal
while(bControlWait)
{
dwWait = WaitForMultipleObjects(NUMEVENTS, m_hEvents, FALSE, INFINITE);
switch(dwWait - WAIT_OBJECT_0)
{
case STOP:
OnStop();
bControlWait = false;
break;
case PAUSE:
OnPause();
ResetEvent(m_hEvents[PAUSE]);
break;
case CONTINUE:
OnContinue();
ResetEvent(m_hEvents[CONTINUE]);
break;
case SHUTDOWN:
OnShutdown();
bControlWait = false;
break;
}
}
return 0;
}
void CService::SetStatus(DWORD dwNewState, DWORD dwNewCheckpoint,
DWORD dwNewHint, DWORD dwNewControls,
DWORD dwExitCode, DWORD dwSpecificExit)
{
// The only state that can set Exit Codes is STOPPED
// Fix if necessary, just in case not set properly.
if(dwNewState != SERVICE_STOPPED)
{
dwExitCode = S_OK;
dwSpecificExit = 0;
}
// Only pending states can set checkpoints or wait hints,
// and pending states *must* set wait hints
if(dwNewState == SERVICE_STOPPED || dwNewState == SERVICE_PAUSED || dwNewState == SERVICE_RUNNING)
{
// Requires hint and checkpoint == 0
// Fix it so that NO_CHANGE from previous state doesn't cause nonzero
dwNewHint = 0;
dwNewCheckpoint = 0;
}
else
{
// Requires hint and checkpoint != 0
if(dwNewHint <= 0 || dwNewCheckpoint <=0)
{
ErrorHandler(_T("CService::SetStatus: Pending statuses require a hint and checkpoint"), true, true, 0);
}
}
// Function can be called by multiple threads - protect member data
EnterCriticalSection(&m_cs);
// Alter states if changing
m_dwState = dwNewState;
if(dwNewCheckpoint != STATE_NO_CHANGE)
m_dwCheckpoint = dwNewCheckpoint;
if(dwNewHint != STATE_NO_CHANGE)
m_dwWaitHint = dwNewHint;
if(dwNewControls != STATE_NO_CHANGE)
m_dwControlsAccepted = dwNewControls;
SERVICE_STATUS ss = { m_dwType, m_dwState, m_dwControlsAccepted,
dwExitCode, dwSpecificExit, m_dwCheckpoint, m_dwWaitHint };
LeaveCriticalSection(&m_cs);
if(!bDebug)
{
if(!SetServiceStatus(m_hServiceStatus, &ss))
ErrorHandler(_T("SetServiceStatus"));
}
}
//Generic error handler which gathers the last error, looks up the description string, and optionally
//prints the string to the event log and/or raises an exception to stop the service.
DWORD CService::ErrorHandler(const TCHAR* psz, bool bPrintEvent, bool bRaiseException, DWORD dwErr)
{
if(bPrintEvent)
{
// CEventLog log(m_szName);
// log.LogWin32Error(0, psz, dwErr);
}
if(bRaiseException)
RaiseException(dwErr, EXCEPTION_NONCONTINUABLE, 0, 0);
return dwErr;
}
void CService::Init()
{}
void CService::OnPause()
{}
void CService::OnContinue()
{}
void CService::OnShutdown()
{}
void CService::HandleUserDefined(DWORD dwControl)
{}
void CService::ParseArgs(DWORD argc, LPTSTR* argv)
{}
BOOL CService::ControlHandler(DWORD dwCtrlType)
{
switch( dwCtrlType )
{
case CTRL_BREAK_EVENT:
case CTRL_C_EVENT:
::SetEvent(m_hEvents[STOP]);
return TRUE;
}
return FALSE;
}