1 /*
2 * PROJECT: ReactOS simple TCP/IP services
3 * LICENSE: GPL - See COPYING in the top level directory
4 * FILE: base/services/tcpsvcs/log.c
5 * PURPOSE: Logging functionality for the service
6 * COPYRIGHT: Copyright 2008 Ged Murphy <gedmurphy@reactos.org>
7 *
8 */
9
10 #include "tcpsvcs.h"
11
12 #define DEBUG
13
14 static LPWSTR lpEventSource = L"tcpsvcs";
15 static WCHAR szLogFileName[MAX_PATH];
16 static HANDLE hLogFile = NULL;
17
18 static OVERLAPPED olWrite;
19
20
21 // needs work
22 static VOID
LogToEventLog(LPCWSTR lpMsg,DWORD errNum,DWORD exitCode,UINT flags)23 LogToEventLog(LPCWSTR lpMsg,
24 DWORD errNum,
25 DWORD exitCode,
26 UINT flags)
27 {
28 HANDLE hEventLog;
29
30 hEventLog = RegisterEventSourceW(NULL, lpEventSource);
31 if (hEventLog)
32 {
33 ReportEventW(hEventLog,
34 (flags & LOG_ERROR) ? EVENTLOG_ERROR_TYPE : EVENTLOG_SUCCESS,
35 0,
36 0,
37 NULL,
38 1,
39 0,
40 &lpMsg,
41 NULL);
42
43 CloseEventLog(hEventLog);
44 }
45 }
46
47 static BOOL
OpenLogFile()48 OpenLogFile()
49 {
50 hLogFile = CreateFileW(szLogFileName,
51 GENERIC_WRITE,
52 FILE_SHARE_READ,
53 NULL,
54 OPEN_ALWAYS,
55 FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED,
56 NULL);
57 if (hLogFile == INVALID_HANDLE_VALUE)
58 {
59 hLogFile = NULL;
60 return FALSE;
61 }
62
63 return TRUE;
64 }
65
66 static VOID
LogToFile(LPCWSTR lpMsg,DWORD errNum,DWORD exitCode,UINT flags)67 LogToFile(LPCWSTR lpMsg,
68 DWORD errNum,
69 DWORD exitCode,
70 UINT flags)
71 {
72 LPWSTR lpFullMsg = NULL;
73 SIZE_T msgLen;
74
75 msgLen = wcslen(lpMsg) + 1;
76
77 if (flags & LOG_ERROR)
78 {
79 LPWSTR lpSysMsg;
80 DWORD eMsgLen;
81
82 eMsgLen = FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
83 NULL,
84 errNum,
85 MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
86 (LPWSTR)&lpSysMsg,
87 0,
88 NULL);
89
90 msgLen = msgLen + eMsgLen + 40;
91
92 lpFullMsg = HeapAlloc(GetProcessHeap(),
93 0,
94 msgLen * sizeof(TCHAR));
95 if (lpFullMsg)
96 {
97 _snwprintf(lpFullMsg,
98 msgLen,
99 L"%s : %s\tErrNum = %lu ExitCode = %lu\r\n",
100 lpMsg,
101 lpSysMsg,
102 errNum,
103 exitCode);
104 }
105
106 LocalFree(lpSysMsg);
107
108 }
109 else
110 {
111 msgLen += 2;
112
113 lpFullMsg = HeapAlloc(GetProcessHeap(),
114 0,
115 msgLen * sizeof(TCHAR));
116 if (lpFullMsg)
117 {
118 _snwprintf(lpFullMsg,
119 msgLen,
120 L"%s\r\n",
121 lpMsg);
122 }
123 }
124
125 /* Make sure the length in bytes doesn't overflow a DWORD */
126 msgLen = wcslen(lpFullMsg);
127 if (msgLen > (MAXDWORD / sizeof(WCHAR)))
128 {
129 RaiseException(EXCEPTION_INT_OVERFLOW, 0, 0, NULL);
130 }
131
132 if (lpFullMsg)
133 {
134 DWORD bytesWritten;
135 DWORD dwRet;
136 BOOL bRet;
137
138 bRet = WriteFile(hLogFile,
139 lpFullMsg,
140 (DWORD)msgLen * sizeof(WCHAR),
141 &bytesWritten,
142 &olWrite);
143 if (!bRet)
144 {
145 if (GetLastError() != ERROR_IO_PENDING)
146 {
147 bRet = FALSE;
148 }
149 else
150 {
151 // Write is pending
152 dwRet = WaitForSingleObject(olWrite.hEvent, INFINITE);
153
154 switch (dwRet)
155 {
156 // event has been signaled
157 case WAIT_OBJECT_0:
158 {
159 bRet = GetOverlappedResult(hLogFile,
160 &olWrite,
161 &bytesWritten,
162 FALSE);
163 break;
164 }
165
166 default:
167 // An error has occurred in WaitForSingleObject.
168 // This usually indicates a problem with the
169 // OVERLAPPED structure's event handle.
170 bRet = FALSE;
171 break;
172 }
173 }
174 }
175
176 if (!bRet || bytesWritten == 0)
177 {
178 LogToEventLog(L"Failed to write to log file",
179 GetLastError(),
180 0,
181 LOG_EVENTLOG | LOG_ERROR);
182 }
183
184 HeapFree(GetProcessHeap(),
185 0,
186 lpFullMsg);
187 }
188
189 if (exitCode > 0)
190 ExitProcess(exitCode);
191 }
192
193
194
195 VOID
LogEvent(LPCWSTR lpMsg,DWORD errNum,DWORD exitCode,UINT flags)196 LogEvent(LPCWSTR lpMsg,
197 DWORD errNum,
198 DWORD exitCode,
199 UINT flags)
200 {
201 #ifdef DEBUG
202 if (flags & LOG_FILE || flags & LOG_ERROR)
203 LogToFile(lpMsg, errNum, exitCode, flags);
204 #endif
205 if (flags & LOG_EVENTLOG)
206 LogToEventLog(lpMsg, errNum, exitCode, flags);
207 }
208
209 BOOL
InitLogging()210 InitLogging()
211 {
212 #ifdef DEBUG
213 BOOL bRet = FALSE;
214
215 if (!GetEnvironmentVariableW(L"SystemDrive", szLogFileName, ARRAYSIZE(szLogFileName)))
216 {
217 StringCchCopyW(szLogFileName, ARRAYSIZE(szLogFileName), L"C:");
218 }
219 StringCchCatW(szLogFileName, ARRAYSIZE(szLogFileName), L"\\tcpsvcs_log.log");
220
221 ZeroMemory(&olWrite, sizeof(OVERLAPPED));
222 olWrite.Offset = 0xFFFFFFFF;
223 olWrite.OffsetHigh = 0xFFFFFFFF;
224 olWrite.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
225 if (olWrite.hEvent)
226 {
227 DeleteFileW(szLogFileName);
228
229 if (OpenLogFile())
230 {
231 WCHAR wcBom = 0xFEFF;
232 DWORD bytesWritten;
233
234 bRet = WriteFile(hLogFile,
235 &wcBom,
236 sizeof(WCHAR),
237 &bytesWritten,
238 &olWrite);
239 if (!bRet)
240 {
241 if (GetLastError() != ERROR_IO_PENDING)
242 {
243 LogToEventLog(L"Failed to write to log file",
244 GetLastError(),
245 0,
246 LOG_EVENTLOG | LOG_ERROR);
247 }
248 else
249 {
250 bRet = TRUE;
251 }
252 }
253 }
254 }
255
256 return bRet;
257 #else
258 return TRUE;
259 #endif
260 }
261
262 VOID
UninitLogging()263 UninitLogging()
264 {
265 if (hLogFile)
266 {
267 FlushFileBuffers(hLogFile);
268 CloseHandle(hLogFile);
269 }
270
271 if (olWrite.hEvent)
272 {
273 CloseHandle(olWrite.hEvent);
274 }
275 }
276