VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/log/log.cpp@ 33540

Last change on this file since 33540 was 33540, checked in by vboxsync, 14 years ago

*: spelling fixes, thanks Timeless!

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 88.3 KB
Line 
1/* $Id: log.cpp 33540 2010-10-28 09:27:05Z vboxsync $ */
2/** @file
3 * Runtime VBox - Logger.
4 */
5
6/*
7 * Copyright (C) 2006-2010 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#include <iprt/log.h>
32#include "internal/iprt.h"
33
34#ifndef IN_RC
35# include <iprt/alloc.h>
36# include <iprt/process.h>
37# include <iprt/semaphore.h>
38# include <iprt/thread.h>
39# include <iprt/mp.h>
40#endif
41#ifdef IN_RING3
42# include <iprt/env.h>
43# include <iprt/file.h>
44# include <iprt/lockvalidator.h>
45# include <iprt/path.h>
46#endif
47#include <iprt/time.h>
48#include <iprt/asm.h>
49#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
50# include <iprt/asm-amd64-x86.h>
51#endif
52#include <iprt/assert.h>
53#include <iprt/err.h>
54#include <iprt/param.h>
55
56#include <iprt/stdarg.h>
57#include <iprt/string.h>
58#include <iprt/ctype.h>
59#ifdef IN_RING3
60# include <iprt/alloca.h>
61# include <stdio.h>
62#endif
63
64
65/*******************************************************************************
66* Structures and Typedefs *
67*******************************************************************************/
68/**
69 * Arguments passed to the output function.
70 */
71typedef struct RTLOGOUTPUTPREFIXEDARGS
72{
73 /** The logger instance. */
74 PRTLOGGER pLogger;
75 /** The flags. (used for prefixing.) */
76 unsigned fFlags;
77 /** The group. (used for prefixing.) */
78 unsigned iGroup;
79} RTLOGOUTPUTPREFIXEDARGS, *PRTLOGOUTPUTPREFIXEDARGS;
80
81
82/*******************************************************************************
83* Internal Functions *
84*******************************************************************************/
85#ifndef IN_RC
86static unsigned rtlogGroupFlags(const char *psz);
87#endif
88#ifdef IN_RING0
89static void rtR0LogLoggerExFallback(uint32_t fDestFlags, uint32_t fFlags, const char *pszFormat, va_list va);
90#endif
91static void rtlogFlush(PRTLOGGER pLogger);
92static DECLCALLBACK(size_t) rtLogOutput(void *pv, const char *pachChars, size_t cbChars);
93static DECLCALLBACK(size_t) rtLogOutputPrefixed(void *pv, const char *pachChars, size_t cbChars);
94
95
96/*******************************************************************************
97* Global Variables *
98*******************************************************************************/
99#ifdef IN_RC
100/** Default logger instance. */
101extern "C" DECLIMPORT(RTLOGGERRC) g_Logger;
102#else /* !IN_RC */
103/** Default logger instance. */
104static PRTLOGGER g_pLogger;
105#endif /* !IN_RC */
106#ifdef IN_RING3
107/** The RTThreadGetWriteLockCount() change caused by the logger mutex semaphore. */
108static uint32_t volatile g_cLoggerLockCount;
109#endif
110#ifdef IN_RING0
111/** Number of per-thread loggers. */
112static int32_t volatile g_cPerThreadLoggers;
113/** Per-thread loggers.
114 * This is just a quick TLS hack suitable for debug logging only.
115 * If we run out of entries, just unload and reload the driver. */
116static struct RTLOGGERPERTHREAD
117{
118 /** The thread. */
119 RTNATIVETHREAD volatile NativeThread;
120 /** The (process / session) key. */
121 uintptr_t volatile uKey;
122 /** The logger instance.*/
123 PRTLOGGER volatile pLogger;
124} g_aPerThreadLoggers[8] =
125{ { NIL_RTNATIVETHREAD, 0, 0},
126 { NIL_RTNATIVETHREAD, 0, 0},
127 { NIL_RTNATIVETHREAD, 0, 0},
128 { NIL_RTNATIVETHREAD, 0, 0},
129 { NIL_RTNATIVETHREAD, 0, 0},
130 { NIL_RTNATIVETHREAD, 0, 0},
131 { NIL_RTNATIVETHREAD, 0, 0},
132 { NIL_RTNATIVETHREAD, 0, 0}
133};
134#endif /* IN_RING0 */
135
136/**
137 * Logger flags instructions.
138 */
139static struct
140{
141 const char *pszInstr; /**< The name */
142 size_t cchInstr; /**< The size of the name. */
143 uint32_t fFlag; /**< The flag value. */
144 bool fInverted; /**< Inverse meaning? */
145}
146const s_aLogFlags[] =
147{
148 { "disabled", sizeof("disabled" ) - 1, RTLOGFLAGS_DISABLED, false },
149 { "enabled", sizeof("enabled" ) - 1, RTLOGFLAGS_DISABLED, true },
150 { "buffered", sizeof("buffered" ) - 1, RTLOGFLAGS_BUFFERED, false },
151 { "unbuffered", sizeof("unbuffered" ) - 1, RTLOGFLAGS_BUFFERED, true },
152 { "usecrlf", sizeof("usecrlf" ) - 1, RTLOGFLAGS_USECRLF, false },
153 { "uself", sizeof("uself" ) - 1, RTLOGFLAGS_USECRLF, true },
154 { "append", sizeof("append" ) - 1, RTLOGFLAGS_APPEND, false },
155 { "overwrite", sizeof("overwrite" ) - 1, RTLOGFLAGS_APPEND, true },
156 { "rel", sizeof("rel" ) - 1, RTLOGFLAGS_REL_TS, false },
157 { "abs", sizeof("abs" ) - 1, RTLOGFLAGS_REL_TS, true },
158 { "dec", sizeof("dec" ) - 1, RTLOGFLAGS_DECIMAL_TS, false },
159 { "hex", sizeof("hex" ) - 1, RTLOGFLAGS_DECIMAL_TS, true },
160 { "writethru", sizeof("writethru" ) - 1, RTLOGFLAGS_WRITE_THROUGH, false },
161 { "writethrough", sizeof("writethrough") - 1, RTLOGFLAGS_WRITE_THROUGH, false },
162 { "flush", sizeof("flush" ) - 1, RTLOGFLAGS_FLUSH, false },
163 { "lockcnts", sizeof("lockcnts" ) - 1, RTLOGFLAGS_PREFIX_LOCK_COUNTS, false },
164 { "cpuid", sizeof("cpuid" ) - 1, RTLOGFLAGS_PREFIX_CPUID, false },
165 { "pid", sizeof("pid" ) - 1, RTLOGFLAGS_PREFIX_PID, false },
166 { "flagno", sizeof("flagno" ) - 1, RTLOGFLAGS_PREFIX_FLAG_NO, false },
167 { "flag", sizeof("flag" ) - 1, RTLOGFLAGS_PREFIX_FLAG, false },
168 { "groupno", sizeof("groupno" ) - 1, RTLOGFLAGS_PREFIX_GROUP_NO, false },
169 { "group", sizeof("group" ) - 1, RTLOGFLAGS_PREFIX_GROUP, false },
170 { "tid", sizeof("tid" ) - 1, RTLOGFLAGS_PREFIX_TID, false },
171 { "thread", sizeof("thread" ) - 1, RTLOGFLAGS_PREFIX_THREAD, false },
172 { "custom", sizeof("custom" ) - 1, RTLOGFLAGS_PREFIX_CUSTOM, false },
173 { "timeprog", sizeof("timeprog" ) - 1, RTLOGFLAGS_PREFIX_TIME_PROG, false },
174 { "time", sizeof("time" ) - 1, RTLOGFLAGS_PREFIX_TIME, false },
175 { "msprog", sizeof("msprog" ) - 1, RTLOGFLAGS_PREFIX_MS_PROG, false },
176 { "tsc", sizeof("tsc" ) - 1, RTLOGFLAGS_PREFIX_TSC, false }, /* before ts! */
177 { "ts", sizeof("ts" ) - 1, RTLOGFLAGS_PREFIX_TS, false },
178};
179
180/**
181 * Logger destination instructions.
182 */
183static struct
184{
185 const char *pszInstr; /**< The name. */
186 size_t cchInstr; /**< The size of the name. */
187 uint32_t fFlag; /**< The corresponding destination flag. */
188} const s_aLogDst[] =
189{
190 { "file", sizeof("file" ) - 1, RTLOGDEST_FILE }, /* Must be 1st! */
191 { "dir", sizeof("dir" ) - 1, RTLOGDEST_FILE }, /* Must be 2nd! */
192 { "stdout", sizeof("stdout" ) - 1, RTLOGDEST_STDOUT },
193 { "stderr", sizeof("stderr" ) - 1, RTLOGDEST_STDERR },
194 { "debugger", sizeof("debugger") - 1, RTLOGDEST_DEBUGGER },
195 { "com", sizeof("com" ) - 1, RTLOGDEST_COM },
196 { "user", sizeof("user" ) - 1, RTLOGDEST_USER },
197};
198
199
200/**
201 * Locks the logger instance.
202 *
203 * @returns See RTSemSpinMutexRequest().
204 * @param pLogger The logger instance.
205 */
206DECLINLINE(int) rtlogLock(PRTLOGGER pLogger)
207{
208#ifndef IN_RC
209 if (pLogger->hSpinMtx != NIL_RTSEMSPINMUTEX)
210 {
211 int rc = RTSemSpinMutexRequest(pLogger->hSpinMtx);
212 if (RT_FAILURE(rc))
213 return rc;
214 }
215#endif
216 return VINF_SUCCESS;
217}
218
219
220/**
221 * Unlocks the logger instance.
222 * @param pLogger The logger instance.
223 */
224DECLINLINE(void) rtlogUnlock(PRTLOGGER pLogger)
225{
226#ifndef IN_RC
227 if (pLogger->hSpinMtx != NIL_RTSEMFASTMUTEX)
228 RTSemSpinMutexRelease(pLogger->hSpinMtx);
229#endif
230 return;
231}
232
233
234#ifndef IN_RC
235/**
236 * Create a logger instance, comprehensive version.
237 *
238 * @returns iprt status code.
239 *
240 * @param ppLogger Where to store the logger instance.
241 * @param fFlags Logger instance flags, a combination of the RTLOGFLAGS_* values.
242 * @param pszGroupSettings The initial group settings.
243 * @param pszEnvVarBase Base name for the environment variables for this instance.
244 * @param cGroups Number of groups in the array.
245 * @param papszGroups Pointer to array of groups. This must stick around for the life of the
246 * logger instance.
247 * @param fDestFlags The destination flags. RTLOGDEST_FILE is ORed if pszFilenameFmt specified.
248 * @param pszErrorMsg A buffer which is filled with an error message if something fails. May be NULL.
249 * @param cchErrorMsg The size of the error message buffer.
250 * @param pszFilenameFmt Log filename format string. Standard RTStrFormat().
251 * @param ... Format arguments.
252 */
253RTDECL(int) RTLogCreateExV(PRTLOGGER *ppLogger, uint32_t fFlags, const char *pszGroupSettings,
254 const char *pszEnvVarBase, unsigned cGroups, const char * const * papszGroups,
255 uint32_t fDestFlags, char *pszErrorMsg, size_t cchErrorMsg, const char *pszFilenameFmt, va_list args)
256{
257 int rc;
258 size_t cb;
259 PRTLOGGER pLogger;
260
261 /*
262 * Validate input.
263 */
264 if ( (cGroups && !papszGroups)
265 || !VALID_PTR(ppLogger)
266 )
267 {
268 AssertMsgFailed(("Invalid parameters!\n"));
269 return VERR_INVALID_PARAMETER;
270 }
271 *ppLogger = NULL;
272
273 if (pszErrorMsg)
274 RTStrPrintf(pszErrorMsg, cchErrorMsg, N_("unknown error"));
275
276 /*
277 * Allocate a logger instance.
278 */
279 cb = RT_OFFSETOF(RTLOGGER, afGroups[cGroups + 1]) + RTPATH_MAX;
280 pLogger = (PRTLOGGER)RTMemAllocZVar(cb);
281 if (pLogger)
282 {
283#if defined(RT_ARCH_X86) && (!defined(LOG_USE_C99) || !defined(RT_WITHOUT_EXEC_ALLOC))
284 uint8_t *pu8Code;
285#endif
286
287 pLogger->u32Magic = RTLOGGER_MAGIC;
288 pLogger->papszGroups = papszGroups;
289 pLogger->cMaxGroups = cGroups;
290 pLogger->cGroups = cGroups;
291 pLogger->pszFilename = (char *)&pLogger->afGroups[cGroups + 1];
292 pLogger->File = NIL_RTFILE;
293 pLogger->fFlags = fFlags;
294 pLogger->fDestFlags = fDestFlags;
295 pLogger->fPendingPrefix = true;
296 if (pszGroupSettings)
297 RTLogGroupSettings(pLogger, pszGroupSettings);
298
299#if defined(RT_ARCH_X86) && (!defined(LOG_USE_C99) || !defined(RT_WITHOUT_EXEC_ALLOC))
300 /*
301 * Emit wrapper code.
302 */
303 pu8Code = (uint8_t *)RTMemExecAlloc(64);
304 if (pu8Code)
305 {
306 pLogger->pfnLogger = *(PFNRTLOGGER*)&pu8Code;
307 *pu8Code++ = 0x68; /* push imm32 */
308 *(void **)pu8Code = pLogger;
309 pu8Code += sizeof(void *);
310 *pu8Code++ = 0xe8; /* call rel32 */
311 *(uint32_t *)pu8Code = (uintptr_t)RTLogLogger - ((uintptr_t)pu8Code + sizeof(uint32_t));
312 pu8Code += sizeof(uint32_t);
313 *pu8Code++ = 0x8d; /* lea esp, [esp + 4] */
314 *pu8Code++ = 0x64;
315 *pu8Code++ = 0x24;
316 *pu8Code++ = 0x04;
317 *pu8Code++ = 0xc3; /* ret near */
318 AssertMsg((uintptr_t)pu8Code - (uintptr_t)pLogger->pfnLogger <= 64,
319 ("Wrapper assembly is too big! %d bytes\n", (uintptr_t)pu8Code - (uintptr_t)pLogger->pfnLogger));
320 rc = VINF_SUCCESS;
321 }
322 else
323 {
324# ifdef RT_OS_LINUX
325 if (pszErrorMsg) /* Most probably SELinux causing trouble since the larger RTMemAlloc succeeded. */
326 RTStrPrintf(pszErrorMsg, cchErrorMsg, N_("mmap(PROT_WRITE | PROT_EXEC) failed -- SELinux?"));
327# endif
328 rc = VERR_NO_MEMORY;
329 }
330 if (RT_SUCCESS(rc))
331#endif /* X86 wrapper code*/
332 {
333#ifdef IN_RING3 /* files and env.vars. are only accessible when in R3 at the present time. */
334 /*
335 * Format the filename.
336 */
337 if (pszFilenameFmt)
338 {
339 RTStrPrintfV(pLogger->pszFilename, RTPATH_MAX, pszFilenameFmt, args);
340 pLogger->fDestFlags |= RTLOGDEST_FILE;
341 }
342
343 /*
344 * Parse the environment variables.
345 */
346 if (pszEnvVarBase)
347 {
348 /* make temp copy of environment variable base. */
349 size_t cchEnvVarBase = strlen(pszEnvVarBase);
350 char *pszEnvVar = (char *)alloca(cchEnvVarBase + 16);
351 memcpy(pszEnvVar, pszEnvVarBase, cchEnvVarBase);
352
353 /*
354 * Destination.
355 */
356 strcpy(pszEnvVar + cchEnvVarBase, "_DEST");
357 const char *pszVar = RTEnvGet(pszEnvVar);
358 if (pszVar)
359 RTLogDestinations(pLogger, pszVar);
360
361 /*
362 * The flags.
363 */
364 strcpy(pszEnvVar + cchEnvVarBase, "_FLAGS");
365 pszVar = RTEnvGet(pszEnvVar);
366 if (pszVar)
367 RTLogFlags(pLogger, pszVar);
368
369 /*
370 * The group settings.
371 */
372 pszEnvVar[cchEnvVarBase] = '\0';
373 pszVar = RTEnvGet(pszEnvVar);
374 if (pszVar)
375 RTLogGroupSettings(pLogger, pszVar);
376 }
377#endif /* IN_RING3 */
378
379 /*
380 * Open the destination(s).
381 */
382 rc = VINF_SUCCESS;
383#ifdef IN_RING3
384 if (pLogger->fDestFlags & RTLOGDEST_FILE)
385 {
386 uint32_t fOpen = RTFILE_O_WRITE | RTFILE_O_DENY_WRITE;
387 if (pLogger->fFlags & RTLOGFLAGS_APPEND)
388 fOpen |= RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND;
389 else
390 fOpen |= RTFILE_O_CREATE_REPLACE;
391 if (pLogger->fFlags & RTLOGFLAGS_WRITE_THROUGH)
392 fOpen |= RTFILE_O_WRITE_THROUGH;
393 rc = RTFileOpen(&pLogger->File, pLogger->pszFilename, fOpen);
394 if (RT_FAILURE(rc) && pszErrorMsg)
395 RTStrPrintf(pszErrorMsg, cchErrorMsg, N_("could not open file '%s' (fOpen=%#x)"), pLogger->pszFilename, fOpen);
396 }
397#endif /* IN_RING3 */
398
399 /*
400 * Create mutex and check how much it counts when entering the lock
401 * so that we can report the values for RTLOGFLAGS_PREFIX_LOCK_COUNTS.
402 */
403 if (RT_SUCCESS(rc))
404 {
405 rc = RTSemSpinMutexCreate(&pLogger->hSpinMtx, RTSEMSPINMUTEX_FLAGS_IRQ_SAFE);
406 if (RT_SUCCESS(rc))
407 {
408#ifdef IN_RING3 /** @todo do counters in ring-0 too? */
409 RTTHREAD Thread = RTThreadSelf();
410 if (Thread != NIL_RTTHREAD)
411 {
412 int32_t c = RTLockValidatorWriteLockGetCount(Thread);
413 RTSemSpinMutexRequest(pLogger->hSpinMtx);
414 c = RTLockValidatorWriteLockGetCount(Thread) - c;
415 RTSemSpinMutexRelease(pLogger->hSpinMtx);
416 ASMAtomicWriteU32(&g_cLoggerLockCount, c);
417 }
418#endif
419 *ppLogger = pLogger;
420 return VINF_SUCCESS;
421 }
422
423 if (pszErrorMsg)
424 RTStrPrintf(pszErrorMsg, cchErrorMsg, N_("failed to create sempahore"));
425 }
426#ifdef IN_RING3
427 RTFileClose(pLogger->File);
428#endif
429#if defined(LOG_USE_C99) && defined(RT_WITHOUT_EXEC_ALLOC)
430 RTMemFree(*(void **)&pLogger->pfnLogger);
431#else
432 RTMemExecFree(*(void **)&pLogger->pfnLogger, 64);
433#endif
434 }
435 RTMemFree(pLogger);
436 }
437 else
438 rc = VERR_NO_MEMORY;
439
440 return rc;
441}
442RT_EXPORT_SYMBOL(RTLogCreateExV);
443
444
445/**
446 * Create a logger instance.
447 *
448 * @returns iprt status code.
449 *
450 * @param ppLogger Where to store the logger instance.
451 * @param fFlags Logger instance flags, a combination of the RTLOGFLAGS_* values.
452 * @param pszGroupSettings The initial group settings.
453 * @param pszEnvVarBase Base name for the environment variables for this instance.
454 * @param cGroups Number of groups in the array.
455 * @param papszGroups Pointer to array of groups. This must stick around for the life of the
456 * logger instance.
457 * @param fDestFlags The destination flags. RTLOGDEST_FILE is ORed if pszFilenameFmt specified.
458 * @param pszFilenameFmt Log filename format string. Standard RTStrFormat().
459 * @param ... Format arguments.
460 */
461RTDECL(int) RTLogCreate(PRTLOGGER *ppLogger, uint32_t fFlags, const char *pszGroupSettings,
462 const char *pszEnvVarBase, unsigned cGroups, const char * const * papszGroups,
463 uint32_t fDestFlags, const char *pszFilenameFmt, ...)
464{
465 va_list args;
466 int rc;
467
468 va_start(args, pszFilenameFmt);
469 rc = RTLogCreateExV(ppLogger, fFlags, pszGroupSettings, pszEnvVarBase, cGroups, papszGroups, fDestFlags, NULL, 0, pszFilenameFmt, args);
470 va_end(args);
471 return rc;
472}
473RT_EXPORT_SYMBOL(RTLogCreate);
474
475
476/**
477 * Create a logger instance.
478 *
479 * @returns iprt status code.
480 *
481 * @param ppLogger Where to store the logger instance.
482 * @param fFlags Logger instance flags, a combination of the RTLOGFLAGS_* values.
483 * @param pszGroupSettings The initial group settings.
484 * @param pszEnvVarBase Base name for the environment variables for this instance.
485 * @param cGroups Number of groups in the array.
486 * @param papszGroups Pointer to array of groups. This must stick around for the life of the
487 * logger instance.
488 * @param fDestFlags The destination flags. RTLOGDEST_FILE is ORed if pszFilenameFmt specified.
489 * @param pszErrorMsg A buffer which is filled with an error message if something fails. May be NULL.
490 * @param cchErrorMsg The size of the error message buffer.
491 * @param pszFilenameFmt Log filename format string. Standard RTStrFormat().
492 * @param ... Format arguments.
493 */
494RTDECL(int) RTLogCreateEx(PRTLOGGER *ppLogger, uint32_t fFlags, const char *pszGroupSettings,
495 const char *pszEnvVarBase, unsigned cGroups, const char * const * papszGroups,
496 uint32_t fDestFlags, char *pszErrorMsg, size_t cchErrorMsg, const char *pszFilenameFmt, ...)
497{
498 va_list args;
499 int rc;
500
501 va_start(args, pszFilenameFmt);
502 rc = RTLogCreateExV(ppLogger, fFlags, pszGroupSettings, pszEnvVarBase, cGroups, papszGroups, fDestFlags, pszErrorMsg, cchErrorMsg, pszFilenameFmt, args);
503 va_end(args);
504 return rc;
505}
506RT_EXPORT_SYMBOL(RTLogCreateEx);
507
508
509/**
510 * Destroys a logger instance.
511 *
512 * The instance is flushed and all output destinations closed (where applicable).
513 *
514 * @returns iprt status code.
515 * @param pLogger The logger instance which close destroyed. NULL is fine.
516 */
517RTDECL(int) RTLogDestroy(PRTLOGGER pLogger)
518{
519 int rc;
520 uint32_t iGroup;
521 RTSEMSPINMUTEX hSpinMtx;
522
523 /*
524 * Validate input.
525 */
526 if (!pLogger)
527 return VINF_SUCCESS;
528 AssertReturn(VALID_PTR(pLogger), VERR_INVALID_POINTER);
529 AssertReturn(pLogger->u32Magic == RTLOGGER_MAGIC, VERR_INVALID_MAGIC);
530
531 /*
532 * Acquire logger instance sem and disable all logging. (paranoia)
533 */
534 rc = rtlogLock(pLogger);
535 AssertMsgRCReturn(rc, ("%Rrc\n", rc), rc);
536
537 pLogger->fFlags |= RTLOGFLAGS_DISABLED;
538 iGroup = pLogger->cGroups;
539 while (iGroup-- > 0)
540 pLogger->afGroups[iGroup] = 0;
541
542 /*
543 * Flush it.
544 */
545 rtlogFlush(pLogger);
546
547 /*
548 * Close output stuffs.
549 */
550#ifdef IN_RING3
551 if (pLogger->File != NIL_RTFILE)
552 {
553 int rc2 = RTFileClose(pLogger->File);
554 AssertRC(rc2);
555 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
556 rc = rc2;
557 pLogger->File = NIL_RTFILE;
558 }
559#endif
560
561 /*
562 * Free the mutex, the wrapper and the instance memory.
563 */
564 hSpinMtx = pLogger->hSpinMtx;
565 pLogger->hSpinMtx = NIL_RTSEMSPINMUTEX;
566 if (hSpinMtx != NIL_RTSEMSPINMUTEX)
567 {
568 int rc2;
569 RTSemSpinMutexRelease(hSpinMtx);
570 rc2 = RTSemSpinMutexDestroy(hSpinMtx);
571 AssertRC(rc2);
572 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
573 rc = rc2;
574 }
575
576 if (pLogger->pfnLogger)
577 {
578#if defined(LOG_USE_C99) && defined(RT_WITHOUT_EXEC_ALLOC)
579 RTMemFree(*(void **)&pLogger->pfnLogger);
580#else
581 RTMemExecFree(*(void **)&pLogger->pfnLogger, 64);
582#endif
583 pLogger->pfnLogger = NULL;
584 }
585 RTMemFree(pLogger);
586
587 return rc;
588}
589RT_EXPORT_SYMBOL(RTLogDestroy);
590
591
592/**
593 * Create a logger instance clone for RC usage.
594 *
595 * @returns iprt status code.
596 *
597 * @param pLogger The logger instance to be cloned.
598 * @param pLoggerRC Where to create the RC logger instance.
599 * @param cbLoggerRC Amount of memory allocated to for the RC logger
600 * instance clone.
601 * @param pfnLoggerRCPtr Pointer to logger wrapper function for this
602 * instance (RC Ptr).
603 * @param pfnFlushRCPtr Pointer to flush function (RC Ptr).
604 * @param fFlags Logger instance flags, a combination of the RTLOGFLAGS_* values.
605 */
606RTDECL(int) RTLogCloneRC(PRTLOGGER pLogger, PRTLOGGERRC pLoggerRC, size_t cbLoggerRC,
607 RTRCPTR pfnLoggerRCPtr, RTRCPTR pfnFlushRCPtr, uint32_t fFlags)
608{
609 /*
610 * Validate input.
611 */
612 if ( !pLoggerRC
613 || !pfnFlushRCPtr
614 || !pfnLoggerRCPtr)
615 {
616 AssertMsgFailed(("Invalid parameters!\n"));
617 return VERR_INVALID_PARAMETER;
618 }
619 if (cbLoggerRC < sizeof(*pLoggerRC))
620 {
621 AssertMsgFailed(("%d min=%d\n", cbLoggerRC, sizeof(*pLoggerRC)));
622 return VERR_INVALID_PARAMETER;
623 }
624
625 /*
626 * Initialize GC instance.
627 */
628 pLoggerRC->offScratch = 0;
629 pLoggerRC->fPendingPrefix = false;
630 pLoggerRC->pfnLogger = pfnLoggerRCPtr;
631 pLoggerRC->pfnFlush = pfnFlushRCPtr;
632 pLoggerRC->u32Magic = RTLOGGERRC_MAGIC;
633 pLoggerRC->fFlags = fFlags | RTLOGFLAGS_DISABLED;
634 pLoggerRC->cGroups = 1;
635 pLoggerRC->afGroups[0] = 0;
636
637 /*
638 * Resolve defaults.
639 */
640 if (!pLogger)
641 {
642 pLogger = RTLogDefaultInstance();
643 if (!pLogger)
644 return VINF_SUCCESS;
645 }
646
647 /*
648 * Check if there's enough space for the groups.
649 */
650 if (cbLoggerRC < (size_t)RT_OFFSETOF(RTLOGGERRC, afGroups[pLogger->cGroups]))
651 {
652 AssertMsgFailed(("%d req=%d cGroups=%d\n", cbLoggerRC, RT_OFFSETOF(RTLOGGERRC, afGroups[pLogger->cGroups]), pLogger->cGroups));
653 return VERR_INVALID_PARAMETER;
654 }
655 memcpy(&pLoggerRC->afGroups[0], &pLogger->afGroups[0], pLogger->cGroups * sizeof(pLoggerRC->afGroups[0]));
656 pLoggerRC->cGroups = pLogger->cGroups;
657
658 /*
659 * Copy bits from the HC instance.
660 */
661 pLoggerRC->fPendingPrefix = pLogger->fPendingPrefix;
662 pLoggerRC->fFlags |= pLogger->fFlags;
663
664 /*
665 * Check if we can remove the disabled flag.
666 */
667 if ( pLogger->fDestFlags
668 && !((pLogger->fFlags | fFlags) & RTLOGFLAGS_DISABLED))
669 pLoggerRC->fFlags &= ~RTLOGFLAGS_DISABLED;
670
671 return VINF_SUCCESS;
672}
673RT_EXPORT_SYMBOL(RTLogCloneRC);
674
675
676/**
677 * Flushes a RC logger instance to a R3 logger.
678 *
679 *
680 * @returns iprt status code.
681 * @param pLogger The R3 logger instance to flush pLoggerRC to. If NULL
682 * the default logger is used.
683 * @param pLoggerRC The RC logger instance to flush.
684 */
685RTDECL(void) RTLogFlushRC(PRTLOGGER pLogger, PRTLOGGERRC pLoggerRC)
686{
687 /*
688 * Resolve defaults.
689 */
690 if (!pLogger)
691 {
692 pLogger = RTLogDefaultInstance();
693 if (!pLogger)
694 {
695 pLoggerRC->offScratch = 0;
696 return;
697 }
698 }
699
700 /*
701 * Any thing to flush?
702 */
703 if ( pLogger->offScratch
704 || pLoggerRC->offScratch)
705 {
706 /*
707 * Acquire logger instance sem.
708 */
709 int rc = rtlogLock(pLogger);
710 if (RT_FAILURE(rc))
711 return;
712
713 /*
714 * Write whatever the GC instance contains to the HC one, and then
715 * flush the HC instance.
716 */
717 if (pLoggerRC->offScratch)
718 {
719 rtLogOutput(pLogger, pLoggerRC->achScratch, pLoggerRC->offScratch);
720 rtLogOutput(pLogger, NULL, 0);
721 pLoggerRC->offScratch = 0;
722 }
723
724 /*
725 * Release the semaphore.
726 */
727 rtlogUnlock(pLogger);
728 }
729}
730RT_EXPORT_SYMBOL(RTLogFlushRC);
731
732
733#ifdef IN_RING3
734/**
735 * Create a logger instance for singled threaded ring-0 usage.
736 *
737 * @returns iprt status code.
738 *
739 * @param pLogger Where to create the logger instance.
740 * @param cbLogger The amount of memory available for the logger instance.
741 * @param pfnLogger Pointer to logger wrapper function for the clone.
742 * @param pfnFlush Pointer to flush function for the clone.
743 * @param fFlags Logger instance flags for the clone, a combination of the RTLOGFLAGS_* values.
744 * @param fDestFlags The destination flags.
745 */
746RTDECL(int) RTLogCreateForR0(PRTLOGGER pLogger, size_t cbLogger, PFNRTLOGGER pfnLogger, PFNRTLOGFLUSH pfnFlush,
747 uint32_t fFlags, uint32_t fDestFlags)
748{
749 /*
750 * Validate input.
751 */
752 AssertPtrReturn(pLogger, VERR_INVALID_PARAMETER);
753 AssertReturn(cbLogger >= sizeof(*pLogger), VERR_INVALID_PARAMETER);
754 AssertReturn(pfnLogger, VERR_INVALID_PARAMETER);
755 AssertReturn(pfnFlush, VERR_INVALID_PARAMETER);
756
757 /*
758 * Initialize the ring-0 instance.
759 */
760 pLogger->offScratch = 0;
761 pLogger->fPendingPrefix = false;
762 pLogger->pfnLogger = pfnLogger;
763 pLogger->pfnFlush = pfnFlush;
764 pLogger->hSpinMtx = NIL_RTSEMSPINMUTEX; /* Not serialized. */
765 pLogger->u32Magic = RTLOGGER_MAGIC;
766 pLogger->fFlags = fFlags;
767 pLogger->fDestFlags = fDestFlags & ~RTLOGDEST_FILE;
768 pLogger->File = NIL_RTFILE;
769 pLogger->pszFilename = NULL;
770 pLogger->papszGroups = NULL;
771 pLogger->cMaxGroups = (uint32_t)((cbLogger - RT_OFFSETOF(RTLOGGER, afGroups[0])) / sizeof(pLogger->afGroups[0]));
772 pLogger->cGroups = 1;
773 pLogger->afGroups[0] = 0;
774 return VINF_SUCCESS;
775}
776RT_EXPORT_SYMBOL(RTLogCreateForR0);
777#endif /* IN_RING3 */
778
779
780/**
781 * Copies the group settings and flags from logger instance to another.
782 *
783 * @returns IPRT status code.
784 * @param pDstLogger The destination logger instance.
785 * @param pSrcLogger The source logger instance. If NULL the default one is used.
786 * @param fFlagsOr OR mask for the flags.
787 * @param fFlagsAnd AND mask for the flags.
788 */
789RTDECL(int) RTLogCopyGroupsAndFlags(PRTLOGGER pDstLogger, PCRTLOGGER pSrcLogger, unsigned fFlagsOr, unsigned fFlagsAnd)
790{
791 int rc;
792 unsigned cGroups;
793
794 /*
795 * Validate input.
796 */
797 AssertPtrReturn(pDstLogger, VERR_INVALID_PARAMETER);
798 AssertPtrNullReturn(pSrcLogger, VERR_INVALID_PARAMETER);
799
800 /*
801 * Resolve defaults.
802 */
803 if (!pSrcLogger)
804 {
805 pSrcLogger = RTLogDefaultInstance();
806 if (!pSrcLogger)
807 {
808 pDstLogger->fFlags |= RTLOGFLAGS_DISABLED;
809 pDstLogger->cGroups = 1;
810 pDstLogger->afGroups[0] = 0;
811 return VINF_SUCCESS;
812 }
813 }
814
815 /*
816 * Copy flags and group settings.
817 */
818 pDstLogger->fFlags = (pSrcLogger->fFlags & fFlagsAnd) | fFlagsOr;
819
820 rc = VINF_SUCCESS;
821 cGroups = pSrcLogger->cGroups;
822 if (cGroups < pDstLogger->cMaxGroups)
823 {
824 AssertMsgFailed(("cMaxGroups=%zd cGroups=%zd (min size %d)\n", pDstLogger->cMaxGroups,
825 pSrcLogger->cGroups, RT_OFFSETOF(RTLOGGER, afGroups[pSrcLogger->cGroups])));
826 rc = VERR_INVALID_PARAMETER;
827 cGroups = pDstLogger->cMaxGroups;
828 }
829 memcpy(&pDstLogger->afGroups[0], &pSrcLogger->afGroups[0], cGroups * sizeof(pDstLogger->afGroups[0]));
830 pDstLogger->cGroups = cGroups;
831
832 return rc;
833}
834RT_EXPORT_SYMBOL(RTLogCopyGroupsAndFlags);
835
836
837/**
838 * Flushes the buffer in one logger instance onto another logger.
839 *
840 * @returns iprt status code.
841 *
842 * @param pSrcLogger The logger instance to flush.
843 * @param pDstLogger The logger instance to flush onto.
844 * If NULL the default logger will be used.
845 */
846RTDECL(void) RTLogFlushToLogger(PRTLOGGER pSrcLogger, PRTLOGGER pDstLogger)
847{
848 /*
849 * Resolve defaults.
850 */
851 if (!pDstLogger)
852 {
853 pDstLogger = RTLogDefaultInstance();
854 if (!pDstLogger)
855 {
856 /* flushing to "/dev/null". */
857 if (pSrcLogger->offScratch)
858 {
859 int rc = rtlogLock(pSrcLogger);
860 if (RT_SUCCESS(rc))
861 {
862 pSrcLogger->offScratch = 0;
863 rtlogLock(pSrcLogger);
864 }
865 }
866 return;
867 }
868 }
869
870 /*
871 * Any thing to flush?
872 */
873 if ( pSrcLogger->offScratch
874 || pDstLogger->offScratch)
875 {
876 /*
877 * Acquire logger semaphores.
878 */
879 int rc = rtlogLock(pDstLogger);
880 if (RT_FAILURE(rc))
881 return;
882 rc = rtlogLock(pSrcLogger);
883 if (RT_SUCCESS(rc))
884 {
885 /*
886 * Write whatever the GC instance contains to the HC one, and then
887 * flush the HC instance.
888 */
889 if (pSrcLogger->offScratch)
890 {
891 rtLogOutput(pDstLogger, pSrcLogger->achScratch, pSrcLogger->offScratch);
892 rtLogOutput(pDstLogger, NULL, 0);
893 pSrcLogger->offScratch = 0;
894 }
895
896 /*
897 * Release the semaphores.
898 */
899 rtlogUnlock(pSrcLogger);
900 }
901 rtlogUnlock(pDstLogger);
902 }
903}
904RT_EXPORT_SYMBOL(RTLogFlushToLogger);
905
906
907/**
908 * Sets the custom prefix callback.
909 *
910 * @returns IPRT status code.
911 * @param pLogger The logger instance.
912 * @param pfnCallback The callback.
913 * @param pvUser The user argument for the callback.
914 * */
915RTDECL(int) RTLogSetCustomPrefixCallback(PRTLOGGER pLogger, PFNRTLOGPREFIX pfnCallback, void *pvUser)
916{
917 /*
918 * Resolve defaults.
919 */
920 if (!pLogger)
921 {
922 pLogger = RTLogDefaultInstance();
923 if (!pLogger)
924 return VINF_SUCCESS;
925 }
926 AssertReturn(pLogger->u32Magic == RTLOGGER_MAGIC, VERR_INVALID_MAGIC);
927
928 /*
929 * Do the work.
930 */
931 rtlogLock(pLogger);
932 pLogger->pvPrefixUserArg = pvUser;
933 pLogger->pfnPrefix = pfnCallback;
934 rtlogUnlock(pLogger);
935
936 return VINF_SUCCESS;
937}
938RT_EXPORT_SYMBOL(RTLogSetCustomPrefixCallback);
939
940
941/**
942 * Matches a group name with a pattern mask in an case insensitive manner (ASCII).
943 *
944 * @returns true if matching and *ppachMask set to the end of the pattern.
945 * @returns false if no match.
946 * @param pszGrp The group name.
947 * @param ppachMask Pointer to the pointer to the mask. Only wildcard supported is '*'.
948 * @param cchMask The length of the mask, including modifiers. The modifiers is why
949 * we update *ppachMask on match.
950 */
951static bool rtlogIsGroupMatching(const char *pszGrp, const char **ppachMask, size_t cchMask)
952{
953 const char *pachMask;
954
955 if (!pszGrp || !*pszGrp)
956 return false;
957 pachMask = *ppachMask;
958 for (;;)
959 {
960 if (RT_C_TO_LOWER(*pszGrp) != RT_C_TO_LOWER(*pachMask))
961 {
962 const char *pszTmp;
963
964 /*
965 * Check for wildcard and do a minimal match if found.
966 */
967 if (*pachMask != '*')
968 return false;
969
970 /* eat '*'s. */
971 do pachMask++;
972 while (--cchMask && *pachMask == '*');
973
974 /* is there more to match? */
975 if ( !cchMask
976 || *pachMask == '.'
977 || *pachMask == '=')
978 break; /* we're good */
979
980 /* do extremely minimal matching (fixme) */
981 pszTmp = strchr(pszGrp, RT_C_TO_LOWER(*pachMask));
982 if (!pszTmp)
983 pszTmp = strchr(pszGrp, RT_C_TO_UPPER(*pachMask));
984 if (!pszTmp)
985 return false;
986 pszGrp = pszTmp;
987 continue;
988 }
989
990 /* done? */
991 if (!*++pszGrp)
992 {
993 /* trailing wildcard is ok. */
994 do
995 {
996 pachMask++;
997 cchMask--;
998 } while (cchMask && *pachMask == '*');
999 if ( !cchMask
1000 || *pachMask == '.'
1001 || *pachMask == '=')
1002 break; /* we're good */
1003 return false;
1004 }
1005
1006 if (!--cchMask)
1007 return false;
1008 pachMask++;
1009 }
1010
1011 /* match */
1012 *ppachMask = pachMask;
1013 return true;
1014}
1015
1016
1017/**
1018 * Updates the group settings for the logger instance using the specified
1019 * specification string.
1020 *
1021 * @returns iprt status code.
1022 * Failures can safely be ignored.
1023 * @param pLogger Logger instance.
1024 * @param pszVar Value to parse.
1025 */
1026RTDECL(int) RTLogGroupSettings(PRTLOGGER pLogger, const char *pszVar)
1027{
1028 /*
1029 * Resolve defaults.
1030 */
1031 if (!pLogger)
1032 {
1033 pLogger = RTLogDefaultInstance();
1034 if (!pLogger)
1035 return VINF_SUCCESS;
1036 }
1037
1038 /*
1039 * Iterate the string.
1040 */
1041 while (*pszVar)
1042 {
1043 /*
1044 * Skip prefixes (blanks, ;, + and -).
1045 */
1046 bool fEnabled = true;
1047 char ch;
1048 const char *pszStart;
1049 unsigned i;
1050 size_t cch;
1051
1052 while ((ch = *pszVar) == '+' || ch == '-' || ch == ' ' || ch == '\t' || ch == '\n' || ch == ';')
1053 {
1054 if (ch == '+' || ch == '-' || ch == ';')
1055 fEnabled = ch != '-';
1056 pszVar++;
1057 }
1058 if (!*pszVar)
1059 break;
1060
1061 /*
1062 * Find end.
1063 */
1064 pszStart = pszVar;
1065 while ((ch = *pszVar) != '\0' && ch != '+' && ch != '-' && ch != ' ' && ch != '\t')
1066 pszVar++;
1067
1068 /*
1069 * Find the group (ascii case insensitive search).
1070 * Special group 'all'.
1071 */
1072 cch = pszVar - pszStart;
1073 if ( cch >= 3
1074 && (pszStart[0] == 'a' || pszStart[0] == 'A')
1075 && (pszStart[1] == 'l' || pszStart[1] == 'L')
1076 && (pszStart[2] == 'l' || pszStart[2] == 'L')
1077 && (cch == 3 || pszStart[3] == '.' || pszStart[3] == '='))
1078 {
1079 /*
1080 * All.
1081 */
1082 unsigned fFlags = cch == 3
1083 ? RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1
1084 : rtlogGroupFlags(&pszStart[3]);
1085 for (i = 0; i < pLogger->cGroups; i++)
1086 {
1087 if (fEnabled)
1088 pLogger->afGroups[i] |= fFlags;
1089 else
1090 pLogger->afGroups[i] &= ~fFlags;
1091 }
1092 }
1093 else
1094 {
1095 /*
1096 * Specific group(s).
1097 */
1098 for (i = 0; i < pLogger->cGroups; i++)
1099 {
1100 const char *psz2 = (const char*)pszStart;
1101 if (rtlogIsGroupMatching(pLogger->papszGroups[i], &psz2, cch))
1102 {
1103 unsigned fFlags = RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1;
1104 if (*psz2 == '.' || *psz2 == '=')
1105 fFlags = rtlogGroupFlags(psz2);
1106 if (fEnabled)
1107 pLogger->afGroups[i] |= fFlags;
1108 else
1109 pLogger->afGroups[i] &= ~fFlags;
1110 }
1111 } /* for each group */
1112 }
1113
1114 } /* parse specification */
1115
1116 return VINF_SUCCESS;
1117}
1118RT_EXPORT_SYMBOL(RTLogGroupSettings);
1119
1120
1121/**
1122 * Interprets the group flags suffix.
1123 *
1124 * @returns Flags specified. (0 is possible!)
1125 * @param psz Start of Suffix. (Either dot or equal sign.)
1126 */
1127static unsigned rtlogGroupFlags(const char *psz)
1128{
1129 unsigned fFlags = 0;
1130
1131 /*
1132 * Literal flags.
1133 */
1134 while (*psz == '.')
1135 {
1136 static struct
1137 {
1138 const char *pszFlag; /* lowercase!! */
1139 unsigned fFlag;
1140 } aFlags[] =
1141 {
1142 { "eo", RTLOGGRPFLAGS_ENABLED },
1143 { "enabledonly",RTLOGGRPFLAGS_ENABLED },
1144 { "e", RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 },
1145 { "enabled", RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 },
1146 { "l1", RTLOGGRPFLAGS_LEVEL_1 },
1147 { "level1", RTLOGGRPFLAGS_LEVEL_1 },
1148 { "l", RTLOGGRPFLAGS_LEVEL_2 },
1149 { "l2", RTLOGGRPFLAGS_LEVEL_2 },
1150 { "level2", RTLOGGRPFLAGS_LEVEL_2 },
1151 { "l3", RTLOGGRPFLAGS_LEVEL_3 },
1152 { "level3", RTLOGGRPFLAGS_LEVEL_3 },
1153 { "l4", RTLOGGRPFLAGS_LEVEL_4 },
1154 { "level4", RTLOGGRPFLAGS_LEVEL_4 },
1155 { "l5", RTLOGGRPFLAGS_LEVEL_5 },
1156 { "level5", RTLOGGRPFLAGS_LEVEL_5 },
1157 { "l6", RTLOGGRPFLAGS_LEVEL_6 },
1158 { "level6", RTLOGGRPFLAGS_LEVEL_6 },
1159 { "f", RTLOGGRPFLAGS_FLOW },
1160 { "flow", RTLOGGRPFLAGS_FLOW },
1161
1162 { "lelik", RTLOGGRPFLAGS_LELIK },
1163 { "michael", RTLOGGRPFLAGS_MICHAEL },
1164 { "dmik", RTLOGGRPFLAGS_DMIK },
1165 { "sunlover", RTLOGGRPFLAGS_SUNLOVER },
1166 { "achim", RTLOGGRPFLAGS_ACHIM },
1167 { "achimha", RTLOGGRPFLAGS_ACHIM },
1168 { "s", RTLOGGRPFLAGS_SANDER },
1169 { "sander", RTLOGGRPFLAGS_SANDER },
1170 { "sandervl", RTLOGGRPFLAGS_SANDER },
1171 { "klaus", RTLOGGRPFLAGS_KLAUS },
1172 { "frank", RTLOGGRPFLAGS_FRANK },
1173 { "b", RTLOGGRPFLAGS_BIRD },
1174 { "bird", RTLOGGRPFLAGS_BIRD },
1175 { "aleksey", RTLOGGRPFLAGS_ALEKSEY },
1176 { "dj", RTLOGGRPFLAGS_DJ },
1177 { "n", RTLOGGRPFLAGS_NONAME },
1178 { "noname", RTLOGGRPFLAGS_NONAME }
1179 };
1180 unsigned i;
1181 bool fFound = false;
1182 psz++;
1183 for (i = 0; i < RT_ELEMENTS(aFlags) && !fFound; i++)
1184 {
1185 const char *psz1 = aFlags[i].pszFlag;
1186 const char *psz2 = psz;
1187 while (*psz1 == RT_C_TO_LOWER(*psz2))
1188 {
1189 psz1++;
1190 psz2++;
1191 if (!*psz1)
1192 {
1193 if ( (*psz2 >= 'a' && *psz2 <= 'z')
1194 || (*psz2 >= 'A' && *psz2 <= 'Z')
1195 || (*psz2 >= '0' && *psz2 <= '9') )
1196 break;
1197 fFlags |= aFlags[i].fFlag;
1198 fFound = true;
1199 psz = psz2;
1200 break;
1201 }
1202 } /* strincmp */
1203 } /* for each flags */
1204 }
1205
1206 /*
1207 * Flag value.
1208 */
1209 if (*psz == '=')
1210 {
1211 psz++;
1212 if (*psz == '~')
1213 fFlags = ~RTStrToInt32(psz + 1);
1214 else
1215 fFlags = RTStrToInt32(psz);
1216 }
1217
1218 return fFlags;
1219}
1220
1221/**
1222 * Helper for RTLogGetGroupSettings.
1223 */
1224static int rtLogGetGroupSettingsAddOne(const char *pszName, uint32_t fGroup, char **ppszBuf, size_t *pcchBuf, bool *pfNotFirst)
1225{
1226#define APPEND_PSZ(psz,cch) do { memcpy(*ppszBuf, (psz), (cch)); *ppszBuf += (cch); *pcchBuf -= (cch); } while (0)
1227#define APPEND_SZ(sz) APPEND_PSZ(sz, sizeof(sz) - 1)
1228#define APPEND_CH(ch) do { **ppszBuf = (ch); *ppszBuf += 1; *pcchBuf -= 1; } while (0)
1229
1230 /*
1231 * Add the name.
1232 */
1233 size_t cchName = strlen(pszName);
1234 if (cchName + 1 + *pfNotFirst > *pcchBuf)
1235 return VERR_BUFFER_OVERFLOW;
1236 if (*pfNotFirst)
1237 APPEND_CH(' ');
1238 else
1239 *pfNotFirst = true;
1240 APPEND_PSZ(pszName, cchName);
1241
1242 /*
1243 * Only generate mnemonics for the simple+common bits.
1244 */
1245 if (fGroup == (RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1))
1246 /* nothing */;
1247 else if ( fGroup == (RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 | RTLOGGRPFLAGS_LEVEL_2 | RTLOGGRPFLAGS_FLOW)
1248 && *pcchBuf >= sizeof(".e.l.f"))
1249 APPEND_SZ(".e.l.f");
1250 else if ( fGroup == (RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 | RTLOGGRPFLAGS_FLOW)
1251 && *pcchBuf >= sizeof(".e.f"))
1252 APPEND_SZ(".e.f");
1253 else if (*pcchBuf >= 1 + 10 + 1)
1254 {
1255 size_t cch;
1256 APPEND_CH('=');
1257 cch = RTStrFormatNumber(*ppszBuf, fGroup, 16, 0, 0, RTSTR_F_SPECIAL | RTSTR_F_32BIT);
1258 *ppszBuf += cch;
1259 *pcchBuf -= cch;
1260 }
1261 else
1262 return VERR_BUFFER_OVERFLOW;
1263
1264#undef APPEND_PSZ
1265#undef APPEND_SZ
1266#undef APPEND_CH
1267 return VINF_SUCCESS;
1268}
1269
1270
1271/**
1272 * Get the current log group settings as a string.
1273 *
1274 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
1275 * @param pLogger Logger instance (NULL for default logger).
1276 * @param pszBuf The output buffer.
1277 * @param cchBuf The size of the output buffer. Must be greater
1278 * than zero.
1279 */
1280RTDECL(int) RTLogGetGroupSettings(PRTLOGGER pLogger, char *pszBuf, size_t cchBuf)
1281{
1282 bool fNotFirst = false;
1283 int rc = VINF_SUCCESS;
1284 uint32_t cGroups;
1285 uint32_t fGroup;
1286 uint32_t i;
1287
1288 Assert(cchBuf);
1289
1290 /*
1291 * Resolve defaults.
1292 */
1293 if (!pLogger)
1294 {
1295 pLogger = RTLogDefaultInstance();
1296 if (!pLogger)
1297 {
1298 *pszBuf = '\0';
1299 return VINF_SUCCESS;
1300 }
1301 }
1302
1303 cGroups = pLogger->cGroups;
1304
1305 /*
1306 * Check if all are the same.
1307 */
1308 fGroup = pLogger->afGroups[0];
1309 for (i = 1; i < cGroups; i++)
1310 if (pLogger->afGroups[i] != fGroup)
1311 break;
1312 if (i >= cGroups)
1313 rc = rtLogGetGroupSettingsAddOne("all", fGroup, &pszBuf, &cchBuf, &fNotFirst);
1314 else
1315 {
1316
1317 /*
1318 * Iterate all the groups and print all that are enabled.
1319 */
1320 for (i = 0; i < cGroups; i++)
1321 {
1322 fGroup = pLogger->afGroups[i];
1323 if (fGroup)
1324 {
1325 const char *pszName = pLogger->papszGroups[i];
1326 if (pszName)
1327 {
1328 rc = rtLogGetGroupSettingsAddOne(pszName, fGroup, &pszBuf, &cchBuf, &fNotFirst);
1329 if (rc)
1330 break;
1331 }
1332 }
1333 }
1334 }
1335
1336 *pszBuf = '\0';
1337 return rc;
1338}
1339RT_EXPORT_SYMBOL(RTLogGetGroupSettings);
1340#endif /* !IN_RC */
1341
1342
1343/**
1344 * Updates the flags for the logger instance using the specified
1345 * specification string.
1346 *
1347 * @returns iprt status code.
1348 * Failures can safely be ignored.
1349 * @param pLogger Logger instance (NULL for default logger).
1350 * @param pszVar Value to parse.
1351 */
1352RTDECL(int) RTLogFlags(PRTLOGGER pLogger, const char *pszVar)
1353{
1354 int rc = VINF_SUCCESS;
1355
1356 /*
1357 * Resolve defaults.
1358 */
1359 if (!pLogger)
1360 {
1361 pLogger = RTLogDefaultInstance();
1362 if (!pLogger)
1363 return VINF_SUCCESS;
1364 }
1365
1366 /*
1367 * Iterate the string.
1368 */
1369 while (*pszVar)
1370 {
1371 /* check no prefix. */
1372 bool fNo = false;
1373 char ch;
1374 unsigned i;
1375
1376 /* skip blanks. */
1377 while (RT_C_IS_SPACE(*pszVar))
1378 pszVar++;
1379 if (!*pszVar)
1380 return rc;
1381
1382 while ((ch = *pszVar) != '\0')
1383 {
1384 if (ch == 'n' && pszVar[1] == 'o')
1385 {
1386 pszVar += 2;
1387 fNo = !fNo;
1388 }
1389 else if (ch == '+')
1390 {
1391 pszVar++;
1392 fNo = true;
1393 }
1394 else if (ch == '-' || ch == '!' || ch == '~')
1395 {
1396 pszVar++;
1397 fNo = !fNo;
1398 }
1399 else
1400 break;
1401 }
1402
1403 /* instruction. */
1404 for (i = 0; i < RT_ELEMENTS(s_aLogFlags); i++)
1405 {
1406 if (!strncmp(pszVar, s_aLogFlags[i].pszInstr, s_aLogFlags[i].cchInstr))
1407 {
1408 if (fNo == s_aLogFlags[i].fInverted)
1409 pLogger->fFlags |= s_aLogFlags[i].fFlag;
1410 else
1411 pLogger->fFlags &= ~s_aLogFlags[i].fFlag;
1412 pszVar += s_aLogFlags[i].cchInstr;
1413 break;
1414 }
1415 }
1416
1417 /* unknown instruction? */
1418 if (i >= RT_ELEMENTS(s_aLogFlags))
1419 {
1420 AssertMsgFailed(("Invalid flags! unknown instruction %.20s\n", pszVar));
1421 pszVar++;
1422 }
1423
1424 /* skip blanks and delimiters. */
1425 while (RT_C_IS_SPACE(*pszVar) || *pszVar == ';')
1426 pszVar++;
1427 } /* while more environment variable value left */
1428
1429 return rc;
1430}
1431RT_EXPORT_SYMBOL(RTLogFlags);
1432
1433
1434/**
1435 * Changes the buffering setting of the specified logger.
1436 *
1437 * This can be used for optimizing longish logging sequences.
1438 *
1439 * @returns The old state.
1440 * @param pLogger The logger instance (NULL is an alias for the
1441 * default logger).
1442 * @param fBuffered The new state.
1443 */
1444RTDECL(bool) RTLogSetBuffering(PRTLOGGER pLogger, bool fBuffered)
1445{
1446 bool fOld;
1447
1448 /*
1449 * Resolve the logger instance.
1450 */
1451 if (!pLogger)
1452 {
1453 pLogger = RTLogDefaultInstance();
1454 if (!pLogger)
1455 return false;
1456 }
1457
1458 rtlogLock(pLogger);
1459 fOld = !!(pLogger->fFlags & RTLOGFLAGS_BUFFERED);
1460 if (fBuffered)
1461 pLogger->fFlags |= RTLOGFLAGS_BUFFERED;
1462 else
1463 pLogger->fFlags &= ~RTLOGFLAGS_BUFFERED;
1464 rtlogUnlock(pLogger);
1465
1466 return fOld;
1467}
1468RT_EXPORT_SYMBOL(RTLogSetBuffering);
1469
1470
1471
1472#ifndef IN_RC
1473/**
1474 * Get the current log flags as a string.
1475 *
1476 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
1477 * @param pLogger Logger instance (NULL for default logger).
1478 * @param pszBuf The output buffer.
1479 * @param cchBuf The size of the output buffer. Must be greater
1480 * than zero.
1481 */
1482RTDECL(int) RTLogGetFlags(PRTLOGGER pLogger, char *pszBuf, size_t cchBuf)
1483{
1484 bool fNotFirst = false;
1485 int rc = VINF_SUCCESS;
1486 uint32_t fFlags;
1487 unsigned i;
1488
1489 Assert(cchBuf);
1490
1491 /*
1492 * Resolve defaults.
1493 */
1494 if (!pLogger)
1495 {
1496 pLogger = RTLogDefaultInstance();
1497 if (!pLogger)
1498 {
1499 *pszBuf = '\0';
1500 return VINF_SUCCESS;
1501 }
1502 }
1503
1504 /*
1505 * Add the flags in the list.
1506 */
1507 fFlags = pLogger->fFlags;
1508 for (i = 0; i < RT_ELEMENTS(s_aLogFlags); i++)
1509 if ( !s_aLogFlags[i].fInverted
1510 ? (s_aLogFlags[i].fFlag & fFlags)
1511 : !(s_aLogFlags[i].fFlag & fFlags))
1512 {
1513 size_t cchInstr = s_aLogFlags[i].cchInstr;
1514 if (cchInstr + fNotFirst + 1 > cchBuf)
1515 {
1516 rc = VERR_BUFFER_OVERFLOW;
1517 break;
1518 }
1519 if (fNotFirst)
1520 {
1521 *pszBuf++ = ' ';
1522 cchBuf--;
1523 }
1524 memcpy(pszBuf, s_aLogFlags[i].pszInstr, cchInstr);
1525 pszBuf += cchInstr;
1526 cchBuf -= cchInstr;
1527 fNotFirst = true;
1528 }
1529 *pszBuf = '\0';
1530 return rc;
1531}
1532RT_EXPORT_SYMBOL(RTLogGetFlags);
1533
1534
1535/**
1536 * Updates the logger destination using the specified string.
1537 *
1538 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
1539 * @param pLogger Logger instance (NULL for default logger).
1540 * @param pszVar The value to parse.
1541 */
1542RTDECL(int) RTLogDestinations(PRTLOGGER pLogger, char const *pszVar)
1543{
1544 /*
1545 * Resolve defaults.
1546 */
1547 if (!pLogger)
1548 {
1549 pLogger = RTLogDefaultInstance();
1550 if (!pLogger)
1551 return VINF_SUCCESS;
1552 }
1553
1554 /*
1555 * Do the parsing.
1556 */
1557 while (*pszVar)
1558 {
1559 bool fNo;
1560 unsigned i;
1561
1562 /* skip blanks. */
1563 while (RT_C_IS_SPACE(*pszVar))
1564 pszVar++;
1565 if (!*pszVar)
1566 break;
1567
1568 /* check no prefix. */
1569 fNo = false;
1570 if (pszVar[0] == 'n' && pszVar[1] == 'o')
1571 {
1572 fNo = true;
1573 pszVar += 2;
1574 }
1575
1576 /* instruction. */
1577 for (i = 0; i < RT_ELEMENTS(s_aLogDst); i++)
1578 {
1579 size_t cchInstr = strlen(s_aLogDst[i].pszInstr);
1580 if (!strncmp(pszVar, s_aLogDst[i].pszInstr, cchInstr))
1581 {
1582 if (!fNo)
1583 pLogger->fDestFlags |= s_aLogDst[i].fFlag;
1584 else
1585 pLogger->fDestFlags &= ~s_aLogDst[i].fFlag;
1586 pszVar += cchInstr;
1587
1588 /* check for value. */
1589 while (RT_C_IS_SPACE(*pszVar))
1590 pszVar++;
1591 if (*pszVar == '=' || *pszVar == ':')
1592 {
1593 const char *pszEnd;
1594
1595 pszVar++;
1596 pszEnd = strchr(pszVar, ';');
1597 if (!pszEnd)
1598 pszEnd = strchr(pszVar, '\0');
1599#ifndef IN_RING0
1600 size_t cch = pszEnd - pszVar;
1601
1602 /* log file name */
1603 if (i == 0 /* file */ && !fNo)
1604 {
1605 AssertReturn(cch < RTPATH_MAX, VERR_OUT_OF_RANGE);
1606 memcpy(pLogger->pszFilename, pszVar, cch);
1607 pLogger->pszFilename[cch] = '\0';
1608 }
1609 /* log directory */
1610 else if (i == 1 /* dir */ && !fNo)
1611 {
1612 char szTmp[RTPATH_MAX];
1613 const char *pszFile = RTPathFilename(pLogger->pszFilename);
1614 size_t cchFile = pszFile ? strlen(pszFile) : 0;
1615 AssertReturn(cchFile + cch + 1 < RTPATH_MAX, VERR_OUT_OF_RANGE);
1616 memcpy(szTmp, cchFile ? pszFile : "", cchFile + 1);
1617
1618 memcpy(pLogger->pszFilename, pszVar, cch);
1619 pLogger->pszFilename[cch] = '\0';
1620 RTPathStripTrailingSlash(pLogger->pszFilename);
1621
1622 cch = strlen(pLogger->pszFilename);
1623 pLogger->pszFilename[cch++] = '/';
1624 memcpy(&pLogger->pszFilename[cch], szTmp, cchFile);
1625 pLogger->pszFilename[cch+cchFile] = '\0';
1626 }
1627 else
1628 AssertMsgFailedReturn(("Invalid destination value! %s%s doesn't take a value!\n",
1629 fNo ? "no" : "", s_aLogDst[i].pszInstr),
1630 VERR_INVALID_PARAMETER);
1631#endif
1632 pszVar = pszEnd + (*pszEnd != '\0');
1633 }
1634 break;
1635 }
1636 }
1637
1638 /* assert known instruction */
1639 AssertMsgReturn(i < RT_ELEMENTS(s_aLogDst),
1640 ("Invalid destination value! unknown instruction %.20s\n", pszVar),
1641 VERR_INVALID_PARAMETER);
1642
1643 /* skip blanks and delimiters. */
1644 while (RT_C_IS_SPACE(*pszVar) || *pszVar == ';')
1645 pszVar++;
1646 } /* while more environment variable value left */
1647
1648 return VINF_SUCCESS;
1649}
1650RT_EXPORT_SYMBOL(RTLogDestinations);
1651
1652
1653/**
1654 * Get the current log destinations as a string.
1655 *
1656 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
1657 * @param pLogger Logger instance (NULL for default logger).
1658 * @param pszBuf The output buffer.
1659 * @param cchBuf The size of the output buffer. Must be greater
1660 * than 0.
1661 */
1662RTDECL(int) RTLogGetDestinations(PRTLOGGER pLogger, char *pszBuf, size_t cchBuf)
1663{
1664 bool fNotFirst = false;
1665 int rc = VINF_SUCCESS;
1666 uint32_t fDestFlags;
1667 unsigned i;
1668
1669 Assert(cchBuf);
1670
1671 /*
1672 * Resolve defaults.
1673 */
1674 if (!pLogger)
1675 {
1676 pLogger = RTLogDefaultInstance();
1677 if (!pLogger)
1678 {
1679 *pszBuf = '\0';
1680 return VINF_SUCCESS;
1681 }
1682 }
1683#define APPEND_PSZ(psz,cch) do { memcpy(pszBuf, (psz), (cch)); pszBuf += (cch); cchBuf -= (cch); } while (0)
1684#define APPEND_SZ(sz) APPEND_PSZ(sz, sizeof(sz) - 1)
1685#define APPEND_CH(ch) do { *pszBuf++ = (ch); cchBuf--; } while (0)
1686
1687 /*
1688 * Add the flags in the list.
1689 */
1690 fDestFlags = pLogger->fDestFlags;
1691 for (i = 2; i < RT_ELEMENTS(s_aLogDst); i++)
1692 if (s_aLogDst[i].fFlag & fDestFlags)
1693 {
1694 size_t cchInstr = s_aLogDst[i].cchInstr;
1695 if (cchInstr + fNotFirst + 1 > cchBuf)
1696 {
1697 rc = VERR_BUFFER_OVERFLOW;
1698 break;
1699 }
1700 if (fNotFirst)
1701 APPEND_CH(' ');
1702 fNotFirst = true;
1703 APPEND_PSZ(s_aLogDst[i].pszInstr, cchInstr);
1704 }
1705
1706 /*
1707 * Add the filename.
1708 */
1709 if ( (fDestFlags & RTLOGDEST_FILE)
1710 && VALID_PTR(pLogger->pszFilename)
1711 && RT_SUCCESS(rc))
1712 {
1713 size_t cchFilename = strlen(pLogger->pszFilename);
1714 if (cchFilename + sizeof("file=") - 1 + fNotFirst + 1 <= cchBuf)
1715 {
1716 if (fNotFirst)
1717 APPEND_SZ(" file=");
1718 else
1719 APPEND_SZ("file=");
1720 APPEND_PSZ(pLogger->pszFilename, cchFilename);
1721 }
1722 else
1723 rc = VERR_BUFFER_OVERFLOW;
1724 }
1725
1726#undef APPEND_PSZ
1727#undef APPEND_SZ
1728#undef APPEND_CH
1729
1730 *pszBuf = '\0';
1731 return rc;
1732}
1733RT_EXPORT_SYMBOL(RTLogGetDestinations);
1734#endif /* !IN_RC */
1735
1736
1737/**
1738 * Flushes the specified logger.
1739 *
1740 * @param pLogger The logger instance to flush.
1741 * If NULL the default instance is used. The default instance
1742 * will not be initialized by this call.
1743 */
1744RTDECL(void) RTLogFlush(PRTLOGGER pLogger)
1745{
1746 /*
1747 * Resolve defaults.
1748 */
1749 if (!pLogger)
1750 {
1751#ifdef IN_RC
1752 pLogger = &g_Logger;
1753#else
1754 pLogger = g_pLogger;
1755#endif
1756 if (!pLogger)
1757 return;
1758 }
1759
1760 /*
1761 * Any thing to flush?
1762 */
1763 if (pLogger->offScratch)
1764 {
1765#ifndef IN_RC
1766 /*
1767 * Acquire logger instance sem.
1768 */
1769 int rc = rtlogLock(pLogger);
1770 if (RT_FAILURE(rc))
1771 return;
1772#endif
1773 /*
1774 * Call worker.
1775 */
1776 rtlogFlush(pLogger);
1777
1778#ifndef IN_RC
1779 /*
1780 * Release the semaphore.
1781 */
1782 rtlogUnlock(pLogger);
1783#endif
1784 }
1785}
1786RT_EXPORT_SYMBOL(RTLogFlush);
1787
1788
1789/**
1790 * Gets the default logger instance, creating it if necessary.
1791 *
1792 * @returns Pointer to default logger instance.
1793 * @returns NULL if no default logger instance available.
1794 */
1795RTDECL(PRTLOGGER) RTLogDefaultInstance(void)
1796{
1797#ifdef IN_RC
1798 return &g_Logger;
1799
1800#else /* !IN_RC */
1801# ifdef IN_RING0
1802 /*
1803 * Check per thread loggers first.
1804 */
1805 if (g_cPerThreadLoggers)
1806 {
1807 const RTNATIVETHREAD Self = RTThreadNativeSelf();
1808 int32_t i = RT_ELEMENTS(g_aPerThreadLoggers);
1809 while (i-- > 0)
1810 if (g_aPerThreadLoggers[i].NativeThread == Self)
1811 return g_aPerThreadLoggers[i].pLogger;
1812 }
1813# endif /* IN_RING0 */
1814
1815 /*
1816 * If no per thread logger, use the default one.
1817 */
1818 if (!g_pLogger)
1819 g_pLogger = RTLogDefaultInit();
1820 return g_pLogger;
1821#endif /* !IN_RC */
1822}
1823RT_EXPORT_SYMBOL(RTLogDefaultInstance);
1824
1825
1826/**
1827 * Gets the default logger instance.
1828 *
1829 * @returns Pointer to default logger instance.
1830 * @returns NULL if no default logger instance available.
1831 */
1832RTDECL(PRTLOGGER) RTLogGetDefaultInstance(void)
1833{
1834#ifdef IN_RC
1835 return &g_Logger;
1836#else
1837# ifdef IN_RING0
1838 /*
1839 * Check per thread loggers first.
1840 */
1841 if (g_cPerThreadLoggers)
1842 {
1843 const RTNATIVETHREAD Self = RTThreadNativeSelf();
1844 int32_t i = RT_ELEMENTS(g_aPerThreadLoggers);
1845 while (i-- > 0)
1846 if (g_aPerThreadLoggers[i].NativeThread == Self)
1847 return g_aPerThreadLoggers[i].pLogger;
1848 }
1849# endif /* IN_RING0 */
1850
1851 return g_pLogger;
1852#endif
1853}
1854RT_EXPORT_SYMBOL(RTLogGetDefaultInstance);
1855
1856
1857#ifndef IN_RC
1858/**
1859 * Sets the default logger instance.
1860 *
1861 * @returns iprt status code.
1862 * @param pLogger The new default logger instance.
1863 */
1864RTDECL(PRTLOGGER) RTLogSetDefaultInstance(PRTLOGGER pLogger)
1865{
1866 return ASMAtomicXchgPtrT(&g_pLogger, pLogger, PRTLOGGER);
1867}
1868RT_EXPORT_SYMBOL(RTLogSetDefaultInstance);
1869#endif /* !IN_RC */
1870
1871
1872#ifdef IN_RING0
1873/**
1874 * Changes the default logger instance for the current thread.
1875 *
1876 * @returns IPRT status code.
1877 * @param pLogger The logger instance. Pass NULL for deregistration.
1878 * @param uKey Associated key for cleanup purposes. If pLogger is NULL,
1879 * all instances with this key will be deregistered. So in
1880 * order to only deregister the instance associated with the
1881 * current thread use 0.
1882 */
1883RTDECL(int) RTLogSetDefaultInstanceThread(PRTLOGGER pLogger, uintptr_t uKey)
1884{
1885 int rc;
1886 RTNATIVETHREAD Self = RTThreadNativeSelf();
1887 if (pLogger)
1888 {
1889 int32_t i;
1890 unsigned j;
1891
1892 AssertReturn(pLogger->u32Magic == RTLOGGER_MAGIC, VERR_INVALID_MAGIC);
1893
1894 /*
1895 * Iterate the table to see if there is already an entry for this thread.
1896 */
1897 i = RT_ELEMENTS(g_aPerThreadLoggers);
1898 while (i-- > 0)
1899 if (g_aPerThreadLoggers[i].NativeThread == Self)
1900 {
1901 ASMAtomicWritePtr((void * volatile *)&g_aPerThreadLoggers[i].uKey, (void *)uKey);
1902 g_aPerThreadLoggers[i].pLogger = pLogger;
1903 return VINF_SUCCESS;
1904 }
1905
1906 /*
1907 * Allocate a new table entry.
1908 */
1909 i = ASMAtomicIncS32(&g_cPerThreadLoggers);
1910 if (i > (int32_t)RT_ELEMENTS(g_aPerThreadLoggers))
1911 {
1912 ASMAtomicDecS32(&g_cPerThreadLoggers);
1913 return VERR_BUFFER_OVERFLOW; /* horrible error code! */
1914 }
1915
1916 for (j = 0; j < 10; j++)
1917 {
1918 i = RT_ELEMENTS(g_aPerThreadLoggers);
1919 while (i-- > 0)
1920 {
1921 AssertCompile(sizeof(RTNATIVETHREAD) == sizeof(void*));
1922 if ( g_aPerThreadLoggers[i].NativeThread == NIL_RTNATIVETHREAD
1923 && ASMAtomicCmpXchgPtr((void * volatile *)&g_aPerThreadLoggers[i].NativeThread, (void *)Self, (void *)NIL_RTNATIVETHREAD))
1924 {
1925 ASMAtomicWritePtr((void * volatile *)&g_aPerThreadLoggers[i].uKey, (void *)uKey);
1926 ASMAtomicWritePtr(&g_aPerThreadLoggers[i].pLogger, pLogger);
1927 return VINF_SUCCESS;
1928 }
1929 }
1930 }
1931
1932 ASMAtomicDecS32(&g_cPerThreadLoggers);
1933 rc = VERR_INTERNAL_ERROR;
1934 }
1935 else
1936 {
1937 /*
1938 * Search the array for the current thread.
1939 */
1940 int32_t i = RT_ELEMENTS(g_aPerThreadLoggers);
1941 while (i-- > 0)
1942 if ( g_aPerThreadLoggers[i].NativeThread == Self
1943 || g_aPerThreadLoggers[i].uKey == uKey)
1944 {
1945 ASMAtomicWriteNullPtr((void * volatile *)&g_aPerThreadLoggers[i].uKey);
1946 ASMAtomicWriteNullPtr(&g_aPerThreadLoggers[i].pLogger);
1947 ASMAtomicWriteHandle(&g_aPerThreadLoggers[i].NativeThread, NIL_RTNATIVETHREAD);
1948 ASMAtomicDecS32(&g_cPerThreadLoggers);
1949 }
1950
1951 rc = VINF_SUCCESS;
1952 }
1953 return rc;
1954}
1955RT_EXPORT_SYMBOL(RTLogSetDefaultInstanceThread);
1956#endif
1957
1958
1959/**
1960 * Write to a logger instance.
1961 *
1962 * @param pLogger Pointer to logger instance.
1963 * @param pszFormat Format string.
1964 * @param args Format arguments.
1965 */
1966RTDECL(void) RTLogLoggerV(PRTLOGGER pLogger, const char *pszFormat, va_list args)
1967{
1968 RTLogLoggerExV(pLogger, 0, ~0U, pszFormat, args);
1969}
1970RT_EXPORT_SYMBOL(RTLogLoggerV);
1971
1972
1973/**
1974 * Write to a logger instance.
1975 *
1976 * This function will check whether the instance, group and flags makes up a
1977 * logging kind which is currently enabled before writing anything to the log.
1978 *
1979 * @param pLogger Pointer to logger instance. If NULL the default logger instance will be attempted.
1980 * @param fFlags The logging flags.
1981 * @param iGroup The group.
1982 * The value ~0U is reserved for compatibility with RTLogLogger[V] and is
1983 * only for internal usage!
1984 * @param pszFormat Format string.
1985 * @param args Format arguments.
1986 */
1987RTDECL(void) RTLogLoggerExV(PRTLOGGER pLogger, unsigned fFlags, unsigned iGroup, const char *pszFormat, va_list args)
1988{
1989 int rc;
1990
1991 /*
1992 * A NULL logger means default instance.
1993 */
1994 if (!pLogger)
1995 {
1996 pLogger = RTLogDefaultInstance();
1997 if (!pLogger)
1998 return;
1999 }
2000
2001 /*
2002 * Validate and correct iGroup.
2003 */
2004 if (iGroup != ~0U && iGroup >= pLogger->cGroups)
2005 iGroup = 0;
2006
2007 /*
2008 * If no output, then just skip it.
2009 */
2010 if ( (pLogger->fFlags & RTLOGFLAGS_DISABLED)
2011#ifndef IN_RC
2012 || !pLogger->fDestFlags
2013#endif
2014 || !pszFormat || !*pszFormat)
2015 return;
2016 if ( iGroup != ~0U
2017 && (pLogger->afGroups[iGroup] & (fFlags | RTLOGGRPFLAGS_ENABLED)) != (fFlags | RTLOGGRPFLAGS_ENABLED))
2018 return;
2019
2020 /*
2021 * Acquire logger instance sem.
2022 */
2023 rc = rtlogLock(pLogger);
2024 if (RT_FAILURE(rc))
2025 {
2026#ifdef IN_RING0
2027 if (pLogger->fDestFlags & ~RTLOGDEST_FILE)
2028 rtR0LogLoggerExFallback(pLogger->fDestFlags, pLogger->fFlags, pszFormat, args);
2029#endif
2030 return;
2031 }
2032
2033 /*
2034 * Format the message and perhaps flush it.
2035 */
2036 if (pLogger->fFlags & (RTLOGFLAGS_PREFIX_MASK | RTLOGFLAGS_USECRLF))
2037 {
2038 RTLOGOUTPUTPREFIXEDARGS OutputArgs;
2039 OutputArgs.pLogger = pLogger;
2040 OutputArgs.iGroup = iGroup;
2041 OutputArgs.fFlags = fFlags;
2042 RTLogFormatV(rtLogOutputPrefixed, &OutputArgs, pszFormat, args);
2043 }
2044 else
2045 RTLogFormatV(rtLogOutput, pLogger, pszFormat, args);
2046 if ( !(pLogger->fFlags & RTLOGFLAGS_BUFFERED)
2047 && pLogger->offScratch)
2048 rtlogFlush(pLogger);
2049
2050 /*
2051 * Release the semaphore.
2052 */
2053 rtlogUnlock(pLogger);
2054}
2055RT_EXPORT_SYMBOL(RTLogLoggerExV);
2056
2057
2058#ifdef IN_RING0
2059/**
2060 * For rtR0LogLoggerExFallbackOutput and rtR0LogLoggerExFallbackFlush.
2061 */
2062typedef struct RTR0LOGLOGGERFALLBACK
2063{
2064 /** The current scratch buffer offset. */
2065 uint32_t offScratch;
2066 /** The destination flags. */
2067 uint32_t fDestFlags;
2068 /** The scratch buffer. */
2069 char achScratch[80];
2070} RTR0LOGLOGGERFALLBACK;
2071/** Pointer to RTR0LOGLOGGERFALLBACK which is used by
2072 * rtR0LogLoggerExFallbackOutput. */
2073typedef RTR0LOGLOGGERFALLBACK *PRTR0LOGLOGGERFALLBACK;
2074
2075
2076/**
2077 * Flushes the fallback buffer.
2078 *
2079 * @param pThis The scratch buffer.
2080 */
2081static void rtR0LogLoggerExFallbackFlush(PRTR0LOGLOGGERFALLBACK pThis)
2082{
2083 if (!pThis->offScratch)
2084 return;
2085
2086 if (pThis->fDestFlags & RTLOGDEST_USER)
2087 RTLogWriteUser(pThis->achScratch, pThis->offScratch);
2088
2089 if (pThis->fDestFlags & RTLOGDEST_DEBUGGER)
2090 RTLogWriteDebugger(pThis->achScratch, pThis->offScratch);
2091
2092 if (pThis->fDestFlags & RTLOGDEST_STDOUT)
2093 RTLogWriteStdOut(pThis->achScratch, pThis->offScratch);
2094
2095 if (pThis->fDestFlags & RTLOGDEST_STDERR)
2096 RTLogWriteStdErr(pThis->achScratch, pThis->offScratch);
2097
2098#ifndef LOG_NO_COM
2099 if (pThis->fDestFlags & RTLOGDEST_COM)
2100 RTLogWriteCom(pThis->achScratch, pThis->offScratch);
2101#endif
2102
2103 /* empty the buffer. */
2104 pThis->offScratch = 0;
2105}
2106
2107
2108/**
2109 * Callback for RTLogFormatV used by rtR0LogLoggerExFallback.
2110 * See PFNLOGOUTPUT() for details.
2111 */
2112static DECLCALLBACK(size_t) rtR0LogLoggerExFallbackOutput(void *pv, const char *pachChars, size_t cbChars)
2113{
2114 PRTR0LOGLOGGERFALLBACK pThis = (PRTR0LOGLOGGERFALLBACK)pv;
2115 if (cbChars)
2116 {
2117 size_t cbRet = 0;
2118 for (;;)
2119 {
2120 /* how much */
2121 uint32_t cb = sizeof(pThis->achScratch) - pThis->offScratch - 1; /* minus 1 - for the string terminator. */
2122 if (cb > cbChars)
2123 cb = (uint32_t)cbChars;
2124
2125 /* copy */
2126 memcpy(&pThis->achScratch[pThis->offScratch], pachChars, cb);
2127
2128 /* advance */
2129 pThis->offScratch += cb;
2130 cbRet += cb;
2131 cbChars -= cb;
2132
2133 /* done? */
2134 if (cbChars <= 0)
2135 return cbRet;
2136
2137 pachChars += cb;
2138
2139 /* flush */
2140 pThis->achScratch[pThis->offScratch] = '\0';
2141 rtR0LogLoggerExFallbackFlush(pThis);
2142 }
2143
2144 /* won't ever get here! */
2145 }
2146 else
2147 {
2148 /*
2149 * Termination call, flush the log.
2150 */
2151 pThis->achScratch[pThis->offScratch] = '\0';
2152 rtR0LogLoggerExFallbackFlush(pThis);
2153 return 0;
2154 }
2155}
2156
2157
2158/**
2159 * Ring-0 fallback for cases where we're unable to grab the lock.
2160 *
2161 * This will happen when we're at a too high IRQL on Windows for instance and
2162 * needs to be dealt with or we'll drop a lot of log output. This fallback will
2163 * only output to some of the log destinations as a few of them may be doing
2164 * dangerous things. We won't be doing any prefixing here either, at least not
2165 * for the present, because it's too much hassle.
2166 *
2167 * @param fDestFlags The destination flags.
2168 * @param fFlags The logger flags.
2169 * @param pszFormat The format string.
2170 * @param va The format arguments.
2171 */
2172static void rtR0LogLoggerExFallback(uint32_t fDestFlags, uint32_t fFlags, const char *pszFormat, va_list va)
2173{
2174 RTR0LOGLOGGERFALLBACK This;
2175 This.fDestFlags = fDestFlags;
2176
2177 /* fallback indicator. */
2178 This.offScratch = 2;
2179 This.achScratch[0] = '[';
2180 This.achScratch[1] = 'F';
2181
2182 /* selected prefixes */
2183 if (fFlags & RTLOGFLAGS_PREFIX_PID)
2184 {
2185 RTPROCESS Process = RTProcSelf();
2186 This.achScratch[This.offScratch++] = ' ';
2187 This.offScratch += RTStrFormatNumber(&This.achScratch[This.offScratch], Process, 16, sizeof(RTPROCESS) * 2, 0, RTSTR_F_ZEROPAD);
2188 }
2189 if (fFlags & RTLOGFLAGS_PREFIX_TID)
2190 {
2191 RTNATIVETHREAD Thread = RTThreadNativeSelf();
2192 This.achScratch[This.offScratch++] = ' ';
2193 This.offScratch += RTStrFormatNumber(&This.achScratch[This.offScratch], Thread, 16, sizeof(RTNATIVETHREAD) * 2, 0, RTSTR_F_ZEROPAD);
2194 }
2195
2196 This.achScratch[This.offScratch++] = ']';
2197 This.achScratch[This.offScratch++] = ' ';
2198
2199 RTLogFormatV(rtR0LogLoggerExFallbackOutput, &This, pszFormat, va);
2200}
2201#endif /* IN_RING0 */
2202
2203
2204/**
2205 * vprintf like function for writing to the default log.
2206 *
2207 * @param pszFormat Printf like format string.
2208 * @param args Optional arguments as specified in pszFormat.
2209 *
2210 * @remark The API doesn't support formatting of floating point numbers at the moment.
2211 */
2212RTDECL(void) RTLogPrintfV(const char *pszFormat, va_list args)
2213{
2214 RTLogLoggerV(NULL, pszFormat, args);
2215}
2216RT_EXPORT_SYMBOL(RTLogPrintfV);
2217
2218
2219/**
2220 * Writes the buffer to the given log device without checking for buffered
2221 * data or anything.
2222 * Used by the RTLogFlush() function.
2223 *
2224 * @param pLogger The logger instance to write to. NULL is not allowed!
2225 */
2226static void rtlogFlush(PRTLOGGER pLogger)
2227{
2228 if (pLogger->offScratch == 0)
2229 return; /* nothing to flush. */
2230
2231#ifndef IN_RC
2232 if (pLogger->fDestFlags & RTLOGDEST_USER)
2233 RTLogWriteUser(pLogger->achScratch, pLogger->offScratch);
2234
2235 if (pLogger->fDestFlags & RTLOGDEST_DEBUGGER)
2236 RTLogWriteDebugger(pLogger->achScratch, pLogger->offScratch);
2237
2238# ifdef IN_RING3
2239 if (pLogger->fDestFlags & RTLOGDEST_FILE)
2240 {
2241 RTFileWrite(pLogger->File, pLogger->achScratch, pLogger->offScratch, NULL);
2242 if (pLogger->fFlags & RTLOGFLAGS_FLUSH)
2243 RTFileFlush(pLogger->File);
2244 }
2245# endif
2246
2247 if (pLogger->fDestFlags & RTLOGDEST_STDOUT)
2248 RTLogWriteStdOut(pLogger->achScratch, pLogger->offScratch);
2249
2250 if (pLogger->fDestFlags & RTLOGDEST_STDERR)
2251 RTLogWriteStdErr(pLogger->achScratch, pLogger->offScratch);
2252
2253# if (defined(IN_RING0) || defined(IN_RC)) && !defined(LOG_NO_COM)
2254 if (pLogger->fDestFlags & RTLOGDEST_COM)
2255 RTLogWriteCom(pLogger->achScratch, pLogger->offScratch);
2256# endif
2257#endif /* !IN_RC */
2258
2259 if (pLogger->pfnFlush)
2260 pLogger->pfnFlush(pLogger);
2261
2262 /* empty the buffer. */
2263 pLogger->offScratch = 0;
2264}
2265
2266
2267/**
2268 * Callback for RTLogFormatV which writes to the com port.
2269 * See PFNLOGOUTPUT() for details.
2270 */
2271static DECLCALLBACK(size_t) rtLogOutput(void *pv, const char *pachChars, size_t cbChars)
2272{
2273 PRTLOGGER pLogger = (PRTLOGGER)pv;
2274 if (cbChars)
2275 {
2276 size_t cbRet = 0;
2277 for (;;)
2278 {
2279#if defined(DEBUG) && defined(IN_RING3)
2280 /* sanity */
2281 if (pLogger->offScratch >= sizeof(pLogger->achScratch))
2282 {
2283 fprintf(stderr, "pLogger->offScratch >= sizeof(pLogger->achScratch) (%#x >= %#x)\n",
2284 pLogger->offScratch, (unsigned)sizeof(pLogger->achScratch));
2285 AssertBreakpoint(); AssertBreakpoint();
2286 }
2287#endif
2288
2289 /* how much */
2290 size_t cb = sizeof(pLogger->achScratch) - pLogger->offScratch - 1;
2291 if (cb > cbChars)
2292 cb = cbChars;
2293
2294 /* copy */
2295 memcpy(&pLogger->achScratch[pLogger->offScratch], pachChars, cb);
2296
2297 /* advance */
2298 pLogger->offScratch += (uint32_t)cb;
2299 cbRet += cb;
2300 cbChars -= cb;
2301
2302 /* done? */
2303 if (cbChars <= 0)
2304 return cbRet;
2305
2306 pachChars += cb;
2307
2308 /* flush */
2309 rtlogFlush(pLogger);
2310 }
2311
2312 /* won't ever get here! */
2313 }
2314 else
2315 {
2316 /*
2317 * Termination call.
2318 * There's always space for a terminator, and it's not counted.
2319 */
2320 pLogger->achScratch[pLogger->offScratch] = '\0';
2321 return 0;
2322 }
2323}
2324
2325
2326
2327/**
2328 * Callback for RTLogFormatV which writes to the logger instance.
2329 * This version supports prefixes.
2330 *
2331 * See PFNLOGOUTPUT() for details.
2332 */
2333static DECLCALLBACK(size_t) rtLogOutputPrefixed(void *pv, const char *pachChars, size_t cbChars)
2334{
2335 PRTLOGOUTPUTPREFIXEDARGS pArgs = (PRTLOGOUTPUTPREFIXEDARGS)pv;
2336 PRTLOGGER pLogger = pArgs->pLogger;
2337 if (cbChars)
2338 {
2339 size_t cbRet = 0;
2340 for (;;)
2341 {
2342 size_t cb = sizeof(pLogger->achScratch) - pLogger->offScratch - 1;
2343 char *psz;
2344 const char *pszNewLine;
2345
2346 /*
2347 * Pending prefix?
2348 */
2349 if (pLogger->fPendingPrefix)
2350 {
2351 pLogger->fPendingPrefix = false;
2352
2353#if defined(DEBUG) && defined(IN_RING3)
2354 /* sanity */
2355 if (pLogger->offScratch >= sizeof(pLogger->achScratch))
2356 {
2357 fprintf(stderr, "pLogger->offScratch >= sizeof(pLogger->achScratch) (%#x >= %#x)\n",
2358 pLogger->offScratch, (unsigned)sizeof(pLogger->achScratch));
2359 AssertBreakpoint(); AssertBreakpoint();
2360 }
2361#endif
2362
2363 /*
2364 * Flush the buffer if there isn't enough room for the maximum prefix config.
2365 * Max is 256, add a couple of extra bytes.
2366 */
2367 if (cb < 256 + 16)
2368 {
2369 rtlogFlush(pLogger);
2370 cb = sizeof(pLogger->achScratch) - pLogger->offScratch - 1;
2371 }
2372
2373 /*
2374 * Write the prefixes.
2375 * psz is pointing to the current position.
2376 */
2377 psz = &pLogger->achScratch[pLogger->offScratch];
2378 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TS)
2379 {
2380 uint64_t u64 = RTTimeNanoTS();
2381 int iBase = 16;
2382 unsigned int fFlags = RTSTR_F_ZEROPAD;
2383 if (pLogger->fFlags & RTLOGFLAGS_DECIMAL_TS)
2384 {
2385 iBase = 10;
2386 fFlags = 0;
2387 }
2388 if (pLogger->fFlags & RTLOGFLAGS_REL_TS)
2389 {
2390 static volatile uint64_t s_u64LastTs;
2391 uint64_t u64DiffTs = u64 - s_u64LastTs;
2392 s_u64LastTs = u64;
2393 /* We could have been preempted just before reading of s_u64LastTs by
2394 * another thread which wrote s_u64LastTs. In that case the difference
2395 * is negative which we simply ignore. */
2396 u64 = (int64_t)u64DiffTs < 0 ? 0 : u64DiffTs;
2397 }
2398 /* 1E15 nanoseconds = 11 days */
2399 psz += RTStrFormatNumber(psz, u64, iBase, 16, 0, fFlags); /* +17 */
2400 *psz++ = ' ';
2401 }
2402 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TSC)
2403 {
2404#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
2405 uint64_t u64 = ASMReadTSC();
2406#else
2407 uint64_t u64 = RTTimeNanoTS();
2408#endif
2409 int iBase = 16;
2410 unsigned int fFlags = RTSTR_F_ZEROPAD;
2411 if (pLogger->fFlags & RTLOGFLAGS_DECIMAL_TS)
2412 {
2413 iBase = 10;
2414 fFlags = 0;
2415 }
2416 if (pLogger->fFlags & RTLOGFLAGS_REL_TS)
2417 {
2418 static volatile uint64_t s_u64LastTsc;
2419 int64_t i64DiffTsc = u64 - s_u64LastTsc;
2420 s_u64LastTsc = u64;
2421 /* We could have been preempted just before reading of s_u64LastTsc by
2422 * another thread which wrote s_u64LastTsc. In that case the difference
2423 * is negative which we simply ignore. */
2424 u64 = i64DiffTsc < 0 ? 0 : i64DiffTsc;
2425 }
2426 /* 1E15 ticks at 4GHz = 69 hours */
2427 psz += RTStrFormatNumber(psz, u64, iBase, 16, 0, fFlags); /* +17 */
2428 *psz++ = ' ';
2429 }
2430 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_MS_PROG)
2431 {
2432#if defined(IN_RING3) || defined(IN_RC)
2433 uint64_t u64 = RTTimeProgramMilliTS();
2434#else
2435 uint64_t u64 = 0;
2436#endif
2437 /* 1E8 milliseconds = 27 hours */
2438 psz += RTStrFormatNumber(psz, u64, 10, 9, 0, RTSTR_F_ZEROPAD);
2439 *psz++ = ' ';
2440 }
2441 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TIME)
2442 {
2443#if defined(IN_RING3) || defined(IN_RING0)
2444 RTTIMESPEC TimeSpec;
2445 RTTIME Time;
2446 RTTimeExplode(&Time, RTTimeNow(&TimeSpec));
2447 psz += RTStrFormatNumber(psz, Time.u8Hour, 10, 2, 0, RTSTR_F_ZEROPAD);
2448 *psz++ = ':';
2449 psz += RTStrFormatNumber(psz, Time.u8Minute, 10, 2, 0, RTSTR_F_ZEROPAD);
2450 *psz++ = ':';
2451 psz += RTStrFormatNumber(psz, Time.u8Second, 10, 2, 0, RTSTR_F_ZEROPAD);
2452 *psz++ = '.';
2453 psz += RTStrFormatNumber(psz, Time.u32Nanosecond / 1000000, 10, 3, 0, RTSTR_F_ZEROPAD);
2454 *psz++ = ' '; /* +17 (3+1+3+1+3+1+4+1) */
2455#else
2456 memset(psz, ' ', 13);
2457 psz += 13;
2458#endif
2459 }
2460 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TIME_PROG)
2461 {
2462#if defined(IN_RING3) || defined(IN_RC)
2463 uint64_t u64 = RTTimeProgramMilliTS();
2464 psz += RTStrFormatNumber(psz, (uint32_t)(u64 / (60 * 60 * 1000)), 10, 2, 0, RTSTR_F_ZEROPAD);
2465 *psz++ = ':';
2466 uint32_t u32 = (uint32_t)(u64 % (60 * 60 * 1000));
2467 psz += RTStrFormatNumber(psz, u32 / (60 * 1000), 10, 2, 0, RTSTR_F_ZEROPAD);
2468 *psz++ = ':';
2469 u32 %= 60 * 1000;
2470 psz += RTStrFormatNumber(psz, u32 / 1000, 10, 2, 0, RTSTR_F_ZEROPAD);
2471 *psz++ = '.';
2472 psz += RTStrFormatNumber(psz, u32 % 1000, 10, 3, 0, RTSTR_F_ZEROPAD);
2473 *psz++ = ' '; /* +20 (9+1+2+1+2+1+3+1) */
2474#else
2475 memset(psz, ' ', 13);
2476 psz += 13;
2477#endif
2478 }
2479# if 0
2480 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_DATETIME)
2481 {
2482 char szDate[32];
2483 RTTIMESPEC Time;
2484 RTTimeSpecToString(RTTimeNow(&Time), szDate, sizeof(szDate));
2485 size_t cch = strlen(szDate);
2486 memcpy(psz, szDate, cch);
2487 psz += cch;
2488 *psz++ = ' '; /* +32 */
2489 }
2490# endif
2491 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_PID)
2492 {
2493#ifndef IN_RC
2494 RTPROCESS Process = RTProcSelf();
2495#else
2496 RTPROCESS Process = NIL_RTPROCESS;
2497#endif
2498 psz += RTStrFormatNumber(psz, Process, 16, sizeof(RTPROCESS) * 2, 0, RTSTR_F_ZEROPAD);
2499 *psz++ = ' '; /* +9 */
2500 }
2501 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TID)
2502 {
2503#ifndef IN_RC
2504 RTNATIVETHREAD Thread = RTThreadNativeSelf();
2505#else
2506 RTNATIVETHREAD Thread = NIL_RTNATIVETHREAD;
2507#endif
2508 psz += RTStrFormatNumber(psz, Thread, 16, sizeof(RTNATIVETHREAD) * 2, 0, RTSTR_F_ZEROPAD);
2509 *psz++ = ' '; /* +17 */
2510 }
2511 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_THREAD)
2512 {
2513#ifdef IN_RING3
2514 const char *pszName = RTThreadSelfName();
2515#elif defined IN_RC
2516 const char *pszName = "EMT-RC";
2517#else
2518 const char *pszName = "R0";
2519#endif
2520 size_t cch = 0;
2521 if (pszName)
2522 {
2523 cch = strlen(pszName);
2524 cch = RT_MIN(cch, 16);
2525 memcpy(psz, pszName, cch);
2526 psz += cch;
2527 }
2528 do
2529 *psz++ = ' ';
2530 while (cch++ < 8); /* +17 */
2531 }
2532 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_CPUID)
2533 {
2534#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
2535 const uint8_t idCpu = ASMGetApicId();
2536#else
2537 const RTCPUID idCpu = RTMpCpuId();
2538#endif
2539 psz += RTStrFormatNumber(psz, idCpu, 16, sizeof(idCpu) * 2, 0, RTSTR_F_ZEROPAD);
2540 *psz++ = ' '; /* +17 */
2541 }
2542#ifndef IN_RC
2543 if ( (pLogger->fFlags & RTLOGFLAGS_PREFIX_CUSTOM)
2544 && pLogger->pfnPrefix)
2545 {
2546 psz += pLogger->pfnPrefix(pLogger, psz, 31, pLogger->pvPrefixUserArg);
2547 *psz++ = ' '; /* +32 */
2548 }
2549#endif
2550 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_LOCK_COUNTS)
2551 {
2552#ifdef IN_RING3 /** @todo implement these counters in ring-0 too? */
2553 RTTHREAD Thread = RTThreadSelf();
2554 if (Thread != NIL_RTTHREAD)
2555 {
2556 uint32_t cReadLocks = RTLockValidatorReadLockGetCount(Thread);
2557 uint32_t cWriteLocks = RTLockValidatorWriteLockGetCount(Thread) - g_cLoggerLockCount;
2558 cReadLocks = RT_MIN(0xfff, cReadLocks);
2559 cWriteLocks = RT_MIN(0xfff, cWriteLocks);
2560 psz += RTStrFormatNumber(psz, cReadLocks, 16, 1, 0, RTSTR_F_ZEROPAD);
2561 *psz++ = '/';
2562 psz += RTStrFormatNumber(psz, cWriteLocks, 16, 1, 0, RTSTR_F_ZEROPAD);
2563 }
2564 else
2565#endif
2566 {
2567 *psz++ = '?';
2568 *psz++ = '/';
2569 *psz++ = '?';
2570 }
2571 *psz++ = ' '; /* +8 */
2572 }
2573 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_FLAG_NO)
2574 {
2575 psz += RTStrFormatNumber(psz, pArgs->fFlags, 16, 8, 0, RTSTR_F_ZEROPAD);
2576 *psz++ = ' '; /* +9 */
2577 }
2578 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_FLAG)
2579 {
2580#ifdef IN_RING3
2581 const char *pszGroup = pArgs->iGroup != ~0U ? pLogger->papszGroups[pArgs->iGroup] : NULL;
2582#else
2583 const char *pszGroup = NULL;
2584#endif
2585 size_t cch = 0;
2586 if (pszGroup)
2587 {
2588 cch = strlen(pszGroup);
2589 cch = RT_MIN(cch, 16);
2590 memcpy(psz, pszGroup, cch);
2591 psz += cch;
2592 }
2593 do
2594 *psz++ = ' ';
2595 while (cch++ < 8); /* +17 */
2596 }
2597 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_GROUP_NO)
2598 {
2599 if (pArgs->iGroup != ~0U)
2600 {
2601 psz += RTStrFormatNumber(psz, pArgs->iGroup, 16, 3, 0, RTSTR_F_ZEROPAD);
2602 *psz++ = ' ';
2603 }
2604 else
2605 {
2606 memcpy(psz, "-1 ", sizeof("-1 ") - 1);
2607 psz += sizeof("-1 ") - 1;
2608 } /* +9 */
2609 }
2610 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_GROUP)
2611 {
2612 const unsigned fGrp = pLogger->afGroups[pArgs->iGroup != ~0U ? pArgs->iGroup : 0];
2613 const char *pszGroup;
2614 size_t cch;
2615 switch (pArgs->fFlags & fGrp)
2616 {
2617 case 0: pszGroup = "--------"; cch = sizeof("--------") - 1; break;
2618 case RTLOGGRPFLAGS_ENABLED: pszGroup = "enabled" ; cch = sizeof("enabled" ) - 1; break;
2619 case RTLOGGRPFLAGS_LEVEL_1: pszGroup = "level 1" ; cch = sizeof("level 1" ) - 1; break;
2620 case RTLOGGRPFLAGS_LEVEL_2: pszGroup = "level 2" ; cch = sizeof("level 2" ) - 1; break;
2621 case RTLOGGRPFLAGS_LEVEL_3: pszGroup = "level 3" ; cch = sizeof("level 3" ) - 1; break;
2622 case RTLOGGRPFLAGS_LEVEL_4: pszGroup = "level 4" ; cch = sizeof("level 4" ) - 1; break;
2623 case RTLOGGRPFLAGS_LEVEL_5: pszGroup = "level 5" ; cch = sizeof("level 5" ) - 1; break;
2624 case RTLOGGRPFLAGS_LEVEL_6: pszGroup = "level 6" ; cch = sizeof("level 6" ) - 1; break;
2625 case RTLOGGRPFLAGS_FLOW: pszGroup = "flow" ; cch = sizeof("flow" ) - 1; break;
2626
2627 /* personal groups */
2628 case RTLOGGRPFLAGS_LELIK: pszGroup = "lelik" ; cch = sizeof("lelik" ) - 1; break;
2629 case RTLOGGRPFLAGS_MICHAEL: pszGroup = "Michael" ; cch = sizeof("Michael" ) - 1; break;
2630 case RTLOGGRPFLAGS_DMIK: pszGroup = "dmik" ; cch = sizeof("dmik" ) - 1; break;
2631 case RTLOGGRPFLAGS_SUNLOVER: pszGroup = "sunlover"; cch = sizeof("sunlover") - 1; break;
2632 case RTLOGGRPFLAGS_ACHIM: pszGroup = "Achim" ; cch = sizeof("Achim" ) - 1; break;
2633 case RTLOGGRPFLAGS_SANDER: pszGroup = "Sander" ; cch = sizeof("Sander" ) - 1; break;
2634 case RTLOGGRPFLAGS_KLAUS: pszGroup = "Klaus" ; cch = sizeof("Klaus" ) - 1; break;
2635 case RTLOGGRPFLAGS_FRANK: pszGroup = "Frank" ; cch = sizeof("Frank" ) - 1; break;
2636 case RTLOGGRPFLAGS_BIRD: pszGroup = "bird" ; cch = sizeof("bird" ) - 1; break;
2637 case RTLOGGRPFLAGS_NONAME: pszGroup = "noname" ; cch = sizeof("noname" ) - 1; break;
2638 default: pszGroup = "????????"; cch = sizeof("????????") - 1; break;
2639 }
2640 if (pszGroup)
2641 {
2642 cch = RT_MIN(cch, 16);
2643 memcpy(psz, pszGroup, cch);
2644 psz += cch;
2645 }
2646 do
2647 *psz++ = ' ';
2648 while (cch++ < 8); /* +17 */
2649 }
2650
2651 /*
2652 * Done, figure what we've used and advance the buffer and free size.
2653 */
2654 cb = psz - &pLogger->achScratch[pLogger->offScratch];
2655 AssertMsg(cb <= 223, ("%#zx (%zd) - fFlags=%#x\n", cb, cb, pLogger->fFlags));
2656 pLogger->offScratch += (uint32_t)cb;
2657 cb = sizeof(pLogger->achScratch) - pLogger->offScratch - 1;
2658 }
2659 else if (cb <= 0)
2660 {
2661 rtlogFlush(pLogger);
2662 cb = sizeof(pLogger->achScratch) - pLogger->offScratch - 1;
2663 }
2664
2665#if defined(DEBUG) && defined(IN_RING3)
2666 /* sanity */
2667 if (pLogger->offScratch >= sizeof(pLogger->achScratch))
2668 {
2669 fprintf(stderr, "pLogger->offScratch >= sizeof(pLogger->achScratch) (%#x >= %#x)\n",
2670 pLogger->offScratch, (unsigned)sizeof(pLogger->achScratch));
2671 AssertBreakpoint(); AssertBreakpoint();
2672 }
2673#endif
2674
2675 /* how much */
2676 if (cb > cbChars)
2677 cb = cbChars;
2678
2679 /* have newline? */
2680 pszNewLine = (const char *)memchr(pachChars, '\n', cb);
2681 if (pszNewLine)
2682 {
2683 if (pLogger->fFlags & RTLOGFLAGS_USECRLF)
2684 cb = pszNewLine - pachChars;
2685 else
2686 {
2687 cb = pszNewLine - pachChars + 1;
2688 pLogger->fPendingPrefix = true;
2689 }
2690 }
2691
2692 /* copy */
2693 memcpy(&pLogger->achScratch[pLogger->offScratch], pachChars, cb);
2694
2695 /* advance */
2696 pLogger->offScratch += (uint32_t)cb;
2697 cbRet += cb;
2698 cbChars -= cb;
2699
2700 if ( pszNewLine
2701 && (pLogger->fFlags & RTLOGFLAGS_USECRLF)
2702 && pLogger->offScratch + 2 < sizeof(pLogger->achScratch))
2703 {
2704 memcpy(&pLogger->achScratch[pLogger->offScratch], "\r\n", 2);
2705 pLogger->offScratch += 2;
2706 cbRet++;
2707 cbChars--;
2708 cb++;
2709 pLogger->fPendingPrefix = true;
2710 }
2711
2712 /* done? */
2713 if (cbChars <= 0)
2714 return cbRet;
2715 pachChars += cb;
2716 }
2717
2718 /* won't ever get here! */
2719 }
2720 else
2721 {
2722 /*
2723 * Termination call.
2724 * There's always space for a terminator, and it's not counted.
2725 */
2726 pLogger->achScratch[pLogger->offScratch] = '\0';
2727 return 0;
2728 }
2729}
2730
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette