OLD | NEW |
1 /* | 1 /* |
2 * This file is part of Adblock Plus <https://adblockplus.org/>, | 2 * This file is part of Adblock Plus <https://adblockplus.org/>, |
3 * Copyright (C) 2006-2016 Eyeo GmbH | 3 * Copyright (C) 2006-2016 Eyeo GmbH |
4 * | 4 * |
5 * Adblock Plus is free software: you can redistribute it and/or modify | 5 * Adblock Plus is free software: you can redistribute it and/or modify |
6 * it under the terms of the GNU General Public License version 3 as | 6 * it under the terms of the GNU General Public License version 3 as |
7 * published by the Free Software Foundation. | 7 * published by the Free Software Foundation. |
8 * | 8 * |
9 * Adblock Plus is distributed in the hope that it will be useful, | 9 * Adblock Plus is distributed in the hope that it will be useful, |
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
12 * GNU General Public License for more details. | 12 * GNU General Public License for more details. |
13 * | 13 * |
14 * You should have received a copy of the GNU General Public License | 14 * You should have received a copy of the GNU General Public License |
15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. | 15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. |
16 */ | 16 */ |
17 | 17 |
18 #include "PluginStdAfx.h" | 18 #include "PluginStdAfx.h" |
19 #include "PluginDebug.h" | 19 #include "PluginDebug.h" |
20 #include "PluginClientBase.h" | 20 #include "ActiveQueue.h" |
21 #include "PluginMutex.h" | 21 #include "PluginMutex.h" |
22 #include "PluginClientBase.h" | |
23 #include "../shared/Utils.h" | 22 #include "../shared/Utils.h" |
| 23 #include <Windows.h> |
24 #include <iomanip> | 24 #include <iomanip> |
25 #include <memory> | 25 #include <memory> |
26 | 26 |
| 27 namespace Trace |
| 28 { |
| 29 Location::Location(int id, int subId) |
| 30 : prefixText() |
| 31 { |
| 32 std::ostringstream ss; |
| 33 ss << "ErrorCode(" << id << "," << subId << ")"; |
| 34 postfixText = ss.str(); |
| 35 }; |
| 36 } |
| 37 |
27 namespace | 38 namespace |
28 { | 39 { |
29 class CPluginDebugLock | 40 class CPluginDebugLock |
30 : public CPluginMutex | 41 : public CPluginMutex |
31 { | 42 { |
32 private: | 43 private: |
33 static CComAutoCriticalSection s_criticalSectionDebugLock; | 44 static CComAutoCriticalSection s_criticalSectionDebugLock; |
34 | 45 |
35 public: | 46 public: |
36 CPluginDebugLock() | 47 CPluginDebugLock() |
37 : CPluginMutex(L"DebugFile", PLUGIN_ERROR_MUTEX_DEBUG_FILE) | 48 : CPluginMutex(L"DebugFile", PLUGIN_ERROR_MUTEX_DEBUG_FILE) |
38 { | 49 { |
39 s_criticalSectionDebugLock.Lock(); | 50 s_criticalSectionDebugLock.Lock(); |
40 } | 51 } |
41 | 52 |
42 ~CPluginDebugLock() | 53 ~CPluginDebugLock() |
43 { | 54 { |
44 s_criticalSectionDebugLock.Unlock(); | 55 s_criticalSectionDebugLock.Unlock(); |
45 } | 56 } |
46 }; | 57 }; |
47 | 58 |
48 CComAutoCriticalSection CPluginDebugLock::s_criticalSectionDebugLock; | 59 CComAutoCriticalSection CPluginDebugLock::s_criticalSectionDebugLock; |
49 | 60 |
| 61 /** |
| 62 * The polymorphic part of a log entry. |
| 63 */ |
50 class LogText | 64 class LogText |
51 { | 65 { |
52 public: | 66 public: |
53 virtual std::string Text() const = 0; | 67 virtual std::string Text() const = 0; |
54 virtual ~LogText() {}; | 68 virtual ~LogText() {}; |
55 }; | 69 }; |
56 | 70 |
57 class LogTextFixed | 71 class LogTextFixed |
58 : public LogText | 72 : public LogText |
59 { | 73 { |
(...skipping 41 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
101 */ | 115 */ |
102 struct SystemTime | 116 struct SystemTime |
103 : public SYSTEMTIME | 117 : public SYSTEMTIME |
104 { | 118 { |
105 SystemTime() | 119 SystemTime() |
106 { | 120 { |
107 ::GetSystemTime(static_cast<SYSTEMTIME*>(this)); | 121 ::GetSystemTime(static_cast<SYSTEMTIME*>(this)); |
108 } | 122 } |
109 }; | 123 }; |
110 | 124 |
| 125 /** |
| 126 * Standard log entry, with variation in the data that generates the log messa
ge. |
| 127 */ |
111 class LogEntry | 128 class LogEntry |
112 { | 129 { |
113 const std::unique_ptr<LogText> text; | 130 std::unique_ptr<LogText> text; |
| 131 |
| 132 const Trace::Location location; |
114 | 133 |
115 std::string InitialPrefix() const | 134 std::string InitialPrefix() const |
116 { | 135 { |
117 std::stringstream ss; | 136 std::stringstream ss; |
118 ss << std::setfill('0') << std::setw(2) << st.wHour; | 137 ss << std::setfill('0') << std::setw(2) << st.wHour; |
119 ss << ":"; | 138 ss << ":"; |
120 ss << std::setfill('0') << std::setw(2) << st.wMinute; | 139 ss << std::setfill('0') << std::setw(2) << st.wMinute; |
121 ss << ":"; | 140 ss << ":"; |
122 ss << std::setfill('0') << std::setw(2) << st.wSecond; | 141 ss << std::setfill('0') << std::setw(2) << st.wSecond; |
123 ss << "."; | 142 ss << "."; |
(...skipping 18 matching lines...) Expand all Loading... |
142 /** | 161 /** |
143 * The process within which the log-generating statement executes. | 162 * The process within which the log-generating statement executes. |
144 */ | 163 */ |
145 const DWORD processId; | 164 const DWORD processId; |
146 | 165 |
147 /** | 166 /** |
148 * The thread within which the log-generating statement executes. | 167 * The thread within which the log-generating statement executes. |
149 */ | 168 */ |
150 const DWORD threadId; | 169 const DWORD threadId; |
151 | 170 |
152 explicit LogEntry(LogText* text) | 171 explicit LogEntry(LogText* text, Trace::Location location) |
153 : processId(::GetCurrentProcessId()), threadId(::GetCurrentThreadId()), te
xt(text) | 172 : text(text), location(location), processId(::GetCurrentProcessId()), thre
adId(::GetCurrentThreadId()) |
154 {} | 173 {} |
155 | 174 |
156 LogEntry(LogText* text, DWORD processId, DWORD threadId) | 175 /** |
157 : processId(processId), threadId(threadId), text(text) | 176 * Move constructor transfers 'text', copies everything else. |
| 177 */ |
| 178 LogEntry(LogEntry&& x) |
| 179 : text(std::move(x.text)), location(x.location), st(x.st), processId(x.pro
cessId), threadId(x.threadId) |
158 {} | 180 {} |
159 | 181 |
| 182 /** |
| 183 * Copy constructor deleted. |
| 184 */ |
| 185 LogEntry(const LogEntry&); // = delete |
| 186 |
| 187 /** |
| 188 * Copy assignment deleted. |
| 189 */ |
| 190 LogEntry& operator=(const LogEntry&); // = delete |
| 191 |
160 void Write(std::ostream& out) const | 192 void Write(std::ostream& out) const |
161 { | 193 { |
162 CPluginDebugLock lock; | 194 CPluginDebugLock lock; |
163 if (lock.IsLocked()) | 195 if (lock.IsLocked()) |
164 { | 196 { |
165 auto lines = text->Text(); | 197 auto lines = text->Text(); |
166 size_t linePosition = 0; | 198 size_t linePosition = 0; |
167 while (true) | 199 while (true) |
168 { | 200 { |
169 auto eolPosition = lines.find('\n', linePosition); | 201 // Standard prefix, timestamp and thread ID |
170 auto prefix = linePosition == 0 ? InitialPrefix() : SubsequentPrefix()
; | 202 auto prefix = linePosition == 0 ? InitialPrefix() : SubsequentPrefix()
; |
171 out << prefix; | 203 out << prefix; |
| 204 |
| 205 // Location prefix, if any |
| 206 if (location.prefixText.length() > 0) |
| 207 { |
| 208 out << location.prefixText << " - "; |
| 209 } |
| 210 |
| 211 // One line of the log text itself |
| 212 auto eolPosition = lines.find('\n', linePosition); |
172 if (eolPosition == std::string::npos) | 213 if (eolPosition == std::string::npos) |
173 { | 214 { |
174 out << lines.substr(linePosition) << "\n"; | 215 out << lines.substr(linePosition); |
175 break; | |
176 } | 216 } |
177 else | 217 else |
178 { | 218 { |
179 out << lines.substr(linePosition, eolPosition - linePosition) << "\n
"; | 219 out << lines.substr(linePosition, eolPosition - linePosition); |
180 linePosition = eolPosition + 1; | 220 linePosition = eolPosition + 1; |
181 } | 221 } |
| 222 |
| 223 // Location postfix, if any |
| 224 if (location.postfixText.length() > 0) |
| 225 { |
| 226 out << " - "; |
| 227 // Hack to deal with how gyp generates file names for Visual Studio |
| 228 if (location.postfixText.find("..\\..\\") == 0) |
| 229 { |
| 230 out << location.postfixText.substr(6); |
| 231 } |
| 232 else |
| 233 { |
| 234 out << location.postfixText; |
| 235 } |
| 236 } |
| 237 out << "\n"; |
| 238 |
| 239 // Check to see if that was the last line |
| 240 if (eolPosition == std::string::npos) |
| 241 { |
| 242 break; |
| 243 } |
182 } | 244 } |
183 out.flush(); | 245 out.flush(); |
184 } | 246 } |
185 } | 247 } |
186 | 248 |
187 void WriteFile(const std::wstring& logFileName) const | 249 void WriteFile(const std::wstring& logFileName) const |
188 { | 250 { |
189 std::ofstream out; | 251 std::ofstream out; |
190 out.open(logFileName, std::ios::app); | 252 out.open(logFileName, std::ios::app); |
191 Write(out); | 253 Write(out); |
(...skipping 10 matching lines...) Expand all Loading... |
202 std::wstring debugFileName = GetDataPath(L"debug_" + std::to_wstring(le.proc
essId) + L".txt"); | 264 std::wstring debugFileName = GetDataPath(L"debug_" + std::to_wstring(le.proc
essId) + L".txt"); |
203 le.WriteFile(debugFileName); | 265 le.WriteFile(debugFileName); |
204 } | 266 } |
205 | 267 |
206 void LogWriteResult(const LogEntry& le) | 268 void LogWriteResult(const LogEntry& le) |
207 { | 269 { |
208 std::wstring debugFileName = GetDataPath(L"debug_result.txt"); | 270 std::wstring debugFileName = GetDataPath(L"debug_result.txt"); |
209 le.WriteFile(debugFileName); | 271 le.WriteFile(debugFileName); |
210 } | 272 } |
211 } | 273 } |
212 | 274 |
| 275 /** |
| 276 * An active queue of 'LogEntry'. |
| 277 * Life cycle is maintained in conjunction with 'LogQueueReference'. |
| 278 */ |
| 279 class LogQueue |
| 280 { |
| 281 /** |
| 282 * Processor for the log queue |
| 283 */ |
| 284 static void ProcessLogEntry(const LogEntry&); |
| 285 |
| 286 /** |
| 287 * Active queue for processing log entries. |
| 288 */ |
| 289 ActiveQueue<LogEntry, decltype(ProcessLogEntry)> queue; |
| 290 |
| 291 public: |
| 292 /** |
| 293 * Insert an item into the queue. |
| 294 */ |
| 295 static void Insert(LogEntry&& t) |
| 296 { |
| 297 /* |
| 298 * Note that this function(indeed, this class) does not have the |
| 299 * responsibility to ensure that an instance of the queue exists. |
| 300 * See 'LogQueueReference' for the class that does. |
| 301 * The non-null guard here is purely defensive; |
| 302 * we shouldn't ever lose queued events because of it. |
| 303 */ |
| 304 if (LogQueueReference::queueMasterReference) |
| 305 { |
| 306 LogQueueReference::queueMasterReference->queue.Insert(std::move(t)); |
| 307 } |
| 308 } |
| 309 |
| 310 LogQueue() |
| 311 : queue(ProcessLogEntry) |
| 312 {} |
| 313 }; |
| 314 |
| 315 /** |
| 316 * The processor for the log queue simply writes each log entry to its correspon
ding log file. |
| 317 * |
| 318 * At present the only log entries that go through the queue also go to the defa
ult log file. |
| 319 * This logic will expand to include other log files as we pass all the log entr
ies through the queue. |
| 320 */ |
| 321 void LogQueue::ProcessLogEntry(const LogEntry& entry) |
| 322 { |
| 323 LogWriteDefault(entry); |
| 324 } |
| 325 |
| 326 /* |
| 327 * Static initializations |
| 328 */ |
| 329 LogQueueReference::pointer LogQueueReference::queueMasterReference; |
| 330 std::mutex LogQueueReference::mutex; |
| 331 |
| 332 /* |
| 333 * Only initialize the instance reference within the constructor body, |
| 334 * so that it's serialized under the protection of the mutex. |
| 335 */ |
| 336 LogQueueReference::LogQueueReference() |
| 337 : queueReference(queueMasterReference) |
| 338 { |
| 339 std::unique_lock<std::mutex> lock(LogQueueReference::mutex); |
| 340 if (!queueMasterReference) |
| 341 { |
| 342 queueMasterReference = pointer(new LogQueue()); |
| 343 LogQueue::Insert(LogEntry(new LogTextFixed("LogQueue start"), HERE_F)); |
| 344 } |
| 345 queueReference = queueMasterReference; |
| 346 } |
| 347 |
| 348 LogQueueReference::~LogQueueReference() |
| 349 { |
| 350 std::unique_lock<std::mutex> lock(LogQueueReference::mutex); |
| 351 queueReference.reset(); |
| 352 if (queueMasterReference.use_count() == 1) |
| 353 { |
| 354 LogQueue::Insert(LogEntry(new LogTextFixed("LogQueue stop"), HERE_F)); |
| 355 queueMasterReference.reset(); |
| 356 } |
| 357 } |
| 358 |
| 359 |
213 #ifdef ENABLE_DEBUG_INFO | 360 #ifdef ENABLE_DEBUG_INFO |
214 | 361 |
215 void CPluginDebug::Debug(const std::string& text) | 362 void CPluginDebug::Debug(const std::string& text) |
216 { | 363 { |
217 LogWriteDefault(LogEntry(new LogTextFixed(text))); | 364 LogWriteDefault(LogEntry(new LogTextFixed(text), Trace::Location())); |
218 } | 365 } |
219 | 366 |
220 void CPluginDebug::Debug(const std::wstring& text) | 367 void CPluginDebug::Debug(const std::wstring& text) |
221 { | 368 { |
222 Debug(ToUtf8String(text)); | 369 Debug(ToUtf8String(text)); |
223 } | 370 } |
224 | 371 |
225 #endif | 372 #endif |
226 | 373 |
227 void CPluginDebug::DebugSystemException(const std::system_error& ex, int errorId
, int errorSubid, const std::string& description) | 374 namespace Trace |
228 { | 375 { |
229 std::string message = description + ", " + ex.code().message() + ", " + ex.wha
t(); | 376 void SystemException(const std::system_error& ex, const std::string& descripti
on, Trace::Location location) |
230 DEBUG_ERROR_LOG(ex.code().value(), errorId, errorSubid, message); | 377 { |
| 378 std::string message = description + ", " + ex.code().message() + ", " + ex.w
hat(); |
| 379 ErrorCode(ex.code().value(), message, location); |
| 380 } |
231 } | 381 } |
232 | 382 |
233 #if (defined ENABLE_DEBUG_INFO) | 383 #if (defined ENABLE_DEBUG_INFO) |
234 | 384 |
235 void CPluginDebug::DebugException(const std::exception& ex) | 385 void CPluginDebug::DebugException(const std::exception& ex, Trace::Location loca
tion) |
236 { | 386 { |
237 auto lt = new LogTextFixed(ex); | 387 auto lt = new LogTextFixed(ex); |
238 LogEntry le(lt); | 388 LogEntry le(lt, location); |
239 #ifdef ENABLE_DEBUG_ERROR | |
240 LogWriteDefault(le); | 389 LogWriteDefault(le); |
241 #endif | |
242 DEBUG_SELFTEST( | |
243 "***************************************************************************
*****\n" | |
244 + lt->text() + "\n" | |
245 "***************************************************************************
*****") | |
246 } | 390 } |
247 | 391 |
248 void CPluginDebug::DebugErrorCode(DWORD errorCode, const std::string& error, DWO
RD processId, DWORD threadId) | 392 void CPluginDebug::DebugErrorCode(DWORD errorCode, const std::string& error, Tra
ce::Location location) |
249 { | 393 { |
250 auto lt = new LogTextErrorCode(errorCode, error); | 394 auto lt = new LogTextErrorCode(errorCode, error); |
251 LogEntry le(lt, processId, threadId); | 395 LogEntry le(lt, location); |
252 #ifdef ENABLE_DEBUG_ERROR | |
253 LogWriteDefault(le); | 396 LogWriteDefault(le); |
254 #endif | |
255 DEBUG_SELFTEST( | |
256 "***************************************************************************
*****\n" | |
257 + lt->text() + "\n" | |
258 "***************************************************************************
*****") | |
259 } | 397 } |
260 | 398 |
261 #endif | 399 #endif |
262 | 400 |
263 // ============================================================================ | 401 // ============================================================================ |
264 // Debug result | 402 // Debug result |
265 // ============================================================================ | 403 // ============================================================================ |
266 | 404 |
267 #ifdef ENABLE_DEBUG_RESULT | 405 #ifdef ENABLE_DEBUG_RESULT |
268 | 406 |
269 void CPluginDebug::DebugResult(const std::wstring& text) | 407 void CPluginDebug::DebugResult(const std::wstring& text) |
270 { | 408 { |
271 LogWriteResult(LogEntry(new LogTextFixed(ToUtf8String(text)))); | 409 LogWriteResult(LogEntry(new LogTextFixed(ToUtf8String(text)), Trace::Location(
))); |
272 } | 410 } |
273 | 411 |
274 void CPluginDebug::DebugResultDomain(const std::wstring& domain) | 412 void CPluginDebug::DebugResultDomain(const std::wstring& domain) |
275 { | 413 { |
276 DebugResult( | 414 DebugResult( |
277 L"==========================================================================
===============\n" | 415 L"==========================================================================
===============\n" |
278 + domain + L"\n" | 416 + domain + L"\n" |
279 L"==========================================================================
==============="); | 417 L"==========================================================================
==============="); |
280 } | 418 } |
281 | 419 |
(...skipping 18 matching lines...) Expand all Loading... |
300 r += s.substr(n - 30, 30); | 438 r += s.substr(n - 30, 30); |
301 return r; | 439 return r; |
302 } | 440 } |
303 } | 441 } |
304 | 442 |
305 void CPluginDebug::DebugResultBlocking(const std::wstring& type, const std::wstr
ing& src, const std::wstring& domain) | 443 void CPluginDebug::DebugResultBlocking(const std::wstring& type, const std::wstr
ing& src, const std::wstring& domain) |
306 { | 444 { |
307 DebugResultFormat(L"Blocked", type, domain.empty() ? L"-" : domain, Shorten(sr
c)); | 445 DebugResultFormat(L"Blocked", type, domain.empty() ? L"-" : domain, Shorten(sr
c)); |
308 } | 446 } |
309 | 447 |
310 | |
311 void CPluginDebug::DebugResultHiding(const std::wstring& tag, const std::wstring
& id, const std::wstring& filter) | 448 void CPluginDebug::DebugResultHiding(const std::wstring& tag, const std::wstring
& id, const std::wstring& filter) |
312 { | 449 { |
313 DebugResultFormat(L"Hidden", tag, L"- " + Shorten(id), filter); | 450 DebugResultFormat(L"Hidden", tag, L"- " + Shorten(id), filter); |
314 } | 451 } |
315 | 452 |
316 #endif // ENABLE_DEBUG_RESULT | 453 #endif // ENABLE_DEBUG_RESULT |
317 | 454 |
318 | |
319 #ifdef ENABLE_DEBUG_RESULT_IGNORED | 455 #ifdef ENABLE_DEBUG_RESULT_IGNORED |
320 | 456 |
321 void CPluginDebug::DebugResultIgnoring(const std::wstring& type, const std::wstr
ing& src, const std::wstring& domain) | 457 void CPluginDebug::DebugResultIgnoring(const std::wstring& type, const std::wstr
ing& src, const std::wstring& domain) |
322 { | 458 { |
323 DebugResultFormat(L"Ignored", type, domain.empty() ? L"-" : domain, Shorten(sr
c)); | 459 DebugResultFormat(L"Ignored", type, domain.empty() ? L"-" : domain, Shorten(sr
c)); |
324 } | 460 } |
325 | 461 |
326 #endif // ENABLE_DEBUG_RESULT_IGNORED | 462 #endif // ENABLE_DEBUG_RESULT_IGNORED |
327 | 463 |
| 464 void Trace::ErrorCode(DWORD errorCode, const std::string& description, Trace::Lo
cation location) |
| 465 { |
| 466 LogQueue::Insert(LogEntry(new LogTextErrorCode(errorCode, description), locati
on)); |
| 467 } |
| 468 |
| 469 void Trace::TextFixed(const std::string& description, Trace::Location location) |
| 470 { |
| 471 LogQueue::Insert(LogEntry(new LogTextFixed(description), location)); |
| 472 } |
| 473 |
328 namespace | 474 namespace |
329 { | 475 { |
330 /* | 476 /* |
331 * To convert a pointer to a hexadecimal number, we need an integral type that
has the same size as that of the pointer. | 477 * To convert a pointer to a hexadecimal number, we need an integral type that
has the same size as that of the pointer. |
332 */ | 478 */ |
333 #if defined(_WIN64) | 479 #if defined(_WIN64) |
334 typedef uint64_t voidIntegral; | 480 typedef uint64_t voidIntegral; |
335 static_assert(sizeof(void*)==sizeof(voidIntegral),"WIN64: sizeof(uint64_t) is
not the same as sizeof(void*)"); | 481 static_assert(sizeof(void*)==sizeof(voidIntegral),"WIN64: sizeof(uint64_t) is
not the same as sizeof(void*)"); |
336 #elif defined(_WIN32) | 482 #elif defined(_WIN32) |
337 typedef uint32_t voidIntegral; | 483 typedef uint32_t voidIntegral; |
338 static_assert(sizeof(void*)==sizeof(voidIntegral),"WIN32: sizeof(uint32_t) is
not the same as sizeof(void*)"); | 484 static_assert(sizeof(void*)==sizeof(voidIntegral),"WIN32: sizeof(uint32_t) is
not the same as sizeof(void*)"); |
339 #else | 485 #else |
340 #error Must compile with either _WIN32 or _WIN64 | 486 #error Must compile with either _WIN32 or _WIN64 |
341 #endif | 487 #endif |
342 } | 488 } |
343 | 489 |
344 std::wstring ToHexLiteral(void const* p) | 490 std::wstring ToHexLiteral(void const* p) |
345 { | 491 { |
346 std::wstringstream ss; | 492 std::wstringstream ss; |
347 ss << L"0x"; | 493 ss << L"0x"; |
348 ss.width(sizeof(p) * 2); | 494 ss.width(sizeof(p) * 2); |
349 ss.fill(L'0'); | 495 ss.fill(L'0'); |
350 ss << std::hex << reinterpret_cast<voidIntegral>(p); | 496 ss << std::hex << reinterpret_cast<voidIntegral>(p); |
351 return ss.str(); | 497 return ss.str(); |
352 } | 498 } |
OLD | NEW |