VirtualBox

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

Last change on this file since 24344 was 23215, checked in by vboxsync, 15 years ago

iprt/log.cpp: RTLOGFLAGS_PREFIX_TIME_PROG for RC.

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