VirtualBox

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

Last change on this file since 88485 was 83657, checked in by vboxsync, 5 years ago

reverted 137105 again as the problem was elsewhere. bugref:9670

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 138.2 KB
Line 
1/* $Id: log.cpp 83657 2020-04-09 17:01:41Z vboxsync $ */
2/** @file
3 * Runtime VBox - Logger.
4 */
5
6/*
7 * Copyright (C) 2006-2020 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* Defined Constants And Macros *
67*********************************************************************************************************************************/
68/** @def RTLOG_RINGBUF_DEFAULT_SIZE
69 * The default ring buffer size. */
70/** @def RTLOG_RINGBUF_MAX_SIZE
71 * The max ring buffer size. */
72/** @def RTLOG_RINGBUF_MIN_SIZE
73 * The min ring buffer size. */
74#ifdef IN_RING0
75# define RTLOG_RINGBUF_DEFAULT_SIZE _64K
76# define RTLOG_RINGBUF_MAX_SIZE _4M
77# define RTLOG_RINGBUF_MIN_SIZE _1K
78#elif defined(IN_RING3) || defined(DOXYGEN_RUNNING)
79# define RTLOG_RINGBUF_DEFAULT_SIZE _512K
80# define RTLOG_RINGBUF_MAX_SIZE _1G
81# define RTLOG_RINGBUF_MIN_SIZE _4K
82#endif
83/** The start of ring buffer eye catcher (16 bytes). */
84#define RTLOG_RINGBUF_EYE_CATCHER "START RING BUF\0"
85AssertCompile(sizeof(RTLOG_RINGBUF_EYE_CATCHER) == 16);
86/** The end of ring buffer eye catcher (16 bytes). This also ensures that the ring buffer
87 * forms are properly terminated C string (leading zero chars). */
88#define RTLOG_RINGBUF_EYE_CATCHER_END "\0\0\0END RING BUF"
89AssertCompile(sizeof(RTLOG_RINGBUF_EYE_CATCHER_END) == 16);
90
91
92/*********************************************************************************************************************************
93* Structures and Typedefs *
94*********************************************************************************************************************************/
95/**
96 * Arguments passed to the output function.
97 */
98typedef struct RTLOGOUTPUTPREFIXEDARGS
99{
100 /** The logger instance. */
101 PRTLOGGER pLogger;
102 /** The flags. (used for prefixing.) */
103 unsigned fFlags;
104 /** The group. (used for prefixing.) */
105 unsigned iGroup;
106} RTLOGOUTPUTPREFIXEDARGS, *PRTLOGOUTPUTPREFIXEDARGS;
107
108#ifndef IN_RC
109
110/**
111 * Internal logger data.
112 *
113 * @remarks Don't make casual changes to this structure.
114 */
115typedef struct RTLOGGERINTERNAL
116{
117 /** The structure revision (RTLOGGERINTERNAL_REV). */
118 uint32_t uRevision;
119 /** The size of the internal logger structure. */
120 uint32_t cbSelf;
121
122 /** Spinning mutex semaphore. Can be NIL. */
123 RTSEMSPINMUTEX hSpinMtx;
124 /** Pointer to the flush function. */
125 PFNRTLOGFLUSH pfnFlush;
126
127 /** Custom prefix callback. */
128 PFNRTLOGPREFIX pfnPrefix;
129 /** Prefix callback argument. */
130 void *pvPrefixUserArg;
131 /** This is set if a prefix is pending. */
132 bool fPendingPrefix;
133 /** Alignment padding. */
134 bool afPadding1[2];
135 /** Set if fully created. Used to avoid confusing in a few functions used to
136 * parse logger settings from environment variables. */
137 bool fCreated;
138
139 /** The max number of groups that there is room for in afGroups and papszGroups.
140 * Used by RTLogCopyGroupAndFlags(). */
141 uint32_t cMaxGroups;
142 /** Pointer to the group name array.
143 * (The data is readonly and provided by the user.) */
144 const char * const *papszGroups;
145
146 /** The number of log entries per group. NULL if
147 * RTLOGFLAGS_RESTRICT_GROUPS is not specified. */
148 uint32_t *pacEntriesPerGroup;
149 /** The max number of entries per group. */
150 uint32_t cMaxEntriesPerGroup;
151
152 /** @name Ring buffer logging
153 * The ring buffer records the last cbRingBuf - 1 of log output. The
154 * other configured log destinations are not touched until someone calls
155 * RTLogFlush(), when the ring buffer content is written to them all.
156 *
157 * The aim here is a fast logging destination, that avoids wasting storage
158 * space saving disk space when dealing with huge log volumes where the
159 * interesting bits usually are found near the end of the log. This is
160 * typically the case for scenarios that crashes or hits assertions.
161 *
162 * RTLogFlush() is called implicitly when hitting an assertion. While on a
163 * crash the most debuggers are able to make calls these days, it's usually
164 * possible to view the ring buffer memory.
165 *
166 * @{ */
167 /** Ring buffer size (including both eye catchers). */
168 uint32_t cbRingBuf;
169 /** Number of bytes passing thru the ring buffer since last RTLogFlush call.
170 * (This is used to avoid writing out the same bytes twice.) */
171 uint64_t volatile cbRingBufUnflushed;
172 /** Ring buffer pointer (points at RTLOG_RINGBUF_EYE_CATCHER). */
173 char *pszRingBuf;
174 /** Current ring buffer position (where to write the next char). */
175 char * volatile pchRingBufCur;
176 /** @} */
177
178 /** Program time base for ring-0 (copy of g_u64ProgramStartNanoTS). */
179 uint64_t nsR0ProgramStart;
180 /** Thread name for use in ring-0 with RTLOGFLAGS_PREFIX_THREAD. */
181 char szR0ThreadName[16];
182
183# ifdef IN_RING3 /* Note! Must be at the end! */
184 /** @name File logging bits for the logger.
185 * @{ */
186 /** Pointer to the function called when starting logging, and when
187 * ending or starting a new log file as part of history rotation.
188 * This can be NULL. */
189 PFNRTLOGPHASE pfnPhase;
190
191 /** Handle to log file (if open). */
192 RTFILE hFile;
193 /** Log file history settings: maximum amount of data to put in a file. */
194 uint64_t cbHistoryFileMax;
195 /** Log file history settings: current amount of data in a file. */
196 uint64_t cbHistoryFileWritten;
197 /** Log file history settings: maximum time to use a file (in seconds). */
198 uint32_t cSecsHistoryTimeSlot;
199 /** Log file history settings: in what time slot was the file created. */
200 uint32_t uHistoryTimeSlotStart;
201 /** Log file history settings: number of older files to keep.
202 * 0 means no history. */
203 uint32_t cHistory;
204 /** Pointer to filename. */
205 char szFilename[RTPATH_MAX];
206 /** @} */
207# endif /* IN_RING3 */
208} RTLOGGERINTERNAL;
209
210/** The revision of the internal logger structure. */
211# define RTLOGGERINTERNAL_REV UINT32_C(11)
212
213# ifdef IN_RING3
214/** The size of the RTLOGGERINTERNAL structure in ring-0. */
215# define RTLOGGERINTERNAL_R0_SIZE RT_UOFFSETOF(RTLOGGERINTERNAL, pfnPhase)
216AssertCompileMemberAlignment(RTLOGGERINTERNAL, hFile, sizeof(void *));
217AssertCompileMemberAlignment(RTLOGGERINTERNAL, cbHistoryFileMax, sizeof(uint64_t));
218# endif
219AssertCompileMemberAlignment(RTLOGGERINTERNAL, cbRingBufUnflushed, sizeof(uint64_t));
220
221#endif /* !IN_RC */
222
223
224/*********************************************************************************************************************************
225* Internal Functions *
226*********************************************************************************************************************************/
227#ifndef IN_RC
228static unsigned rtlogGroupFlags(const char *psz);
229#endif
230#ifdef IN_RING0
231static void rtR0LogLoggerExFallback(uint32_t fDestFlags, uint32_t fFlags, PRTLOGGERINTERNAL pInt,
232 const char *pszFormat, va_list va);
233#endif
234#ifdef IN_RING3
235static int rtR3LogOpenFileDestination(PRTLOGGER pLogger, PRTERRINFO pErrInfo);
236#endif
237#ifndef IN_RC
238static void rtLogRingBufFlush(PRTLOGGER pLogger);
239#endif
240static void rtlogFlush(PRTLOGGER pLogger, bool fNeedSpace);
241static DECLCALLBACK(size_t) rtLogOutput(void *pv, const char *pachChars, size_t cbChars);
242static DECLCALLBACK(size_t) rtLogOutputPrefixed(void *pv, const char *pachChars, size_t cbChars);
243static void rtlogLoggerExVLocked(PRTLOGGER pLogger, unsigned fFlags, unsigned iGroup, const char *pszFormat, va_list args);
244#ifndef IN_RC
245static void rtlogLoggerExFLocked(PRTLOGGER pLogger, unsigned fFlags, unsigned iGroup, const char *pszFormat, ...);
246#endif
247
248
249/*********************************************************************************************************************************
250* Global Variables *
251*********************************************************************************************************************************/
252#ifdef IN_RC
253/** Default logger instance. Make it weak because our RC module loader does not
254 * necessarily resolve this symbol and the compiler _must_ check if this is
255 * the case or not. That doesn't work for Darwin (``incompatible feature used:
256 * .weak_reference (must specify "-dynamic" to be used'') */
257# ifdef RT_OS_DARWIN
258extern "C" DECLIMPORT(RTLOGGERRC) g_Logger;
259# else
260extern "C" DECLWEAK(DECLIMPORT(RTLOGGERRC)) g_Logger;
261# endif
262#else /* !IN_RC */
263/** Default logger instance. */
264static PRTLOGGER g_pLogger;
265#endif /* !IN_RC */
266#ifdef IN_RING3
267/** The RTThreadGetWriteLockCount() change caused by the logger mutex semaphore. */
268static uint32_t volatile g_cLoggerLockCount;
269#endif
270
271#ifdef IN_RING0
272/** Number of per-thread loggers. */
273static int32_t volatile g_cPerThreadLoggers;
274/** Per-thread loggers.
275 * This is just a quick TLS hack suitable for debug logging only.
276 * If we run out of entries, just unload and reload the driver. */
277static struct RTLOGGERPERTHREAD
278{
279 /** The thread. */
280 RTNATIVETHREAD volatile NativeThread;
281 /** The (process / session) key. */
282 uintptr_t volatile uKey;
283 /** The logger instance.*/
284 PRTLOGGER volatile pLogger;
285} g_aPerThreadLoggers[8] =
286{
287 { NIL_RTNATIVETHREAD, 0, 0},
288 { NIL_RTNATIVETHREAD, 0, 0},
289 { NIL_RTNATIVETHREAD, 0, 0},
290 { NIL_RTNATIVETHREAD, 0, 0},
291 { NIL_RTNATIVETHREAD, 0, 0},
292 { NIL_RTNATIVETHREAD, 0, 0},
293 { NIL_RTNATIVETHREAD, 0, 0},
294 { NIL_RTNATIVETHREAD, 0, 0}
295};
296#endif /* IN_RING0 */
297
298/**
299 * Logger flags instructions.
300 */
301static struct
302{
303 const char *pszInstr; /**< The name */
304 size_t cchInstr; /**< The size of the name. */
305 uint32_t fFlag; /**< The flag value. */
306 bool fInverted; /**< Inverse meaning? */
307 uint32_t fFixedDest; /**< RTLOGDEST_FIXED_XXX flags blocking this. */
308} const g_aLogFlags[] =
309{
310 { "disabled", sizeof("disabled" ) - 1, RTLOGFLAGS_DISABLED, false, 0 },
311 { "enabled", sizeof("enabled" ) - 1, RTLOGFLAGS_DISABLED, true, 0 },
312 { "buffered", sizeof("buffered" ) - 1, RTLOGFLAGS_BUFFERED, false, 0 },
313 { "unbuffered", sizeof("unbuffered" ) - 1, RTLOGFLAGS_BUFFERED, true, 0 },
314 { "usecrlf", sizeof("usecrlf" ) - 1, RTLOGFLAGS_USECRLF, false, 0 },
315 { "uself", sizeof("uself" ) - 1, RTLOGFLAGS_USECRLF, true, 0 },
316 { "append", sizeof("append" ) - 1, RTLOGFLAGS_APPEND, false, RTLOGDEST_FIXED_FILE },
317 { "overwrite", sizeof("overwrite" ) - 1, RTLOGFLAGS_APPEND, true, RTLOGDEST_FIXED_FILE },
318 { "rel", sizeof("rel" ) - 1, RTLOGFLAGS_REL_TS, false, 0 },
319 { "abs", sizeof("abs" ) - 1, RTLOGFLAGS_REL_TS, true, 0 },
320 { "dec", sizeof("dec" ) - 1, RTLOGFLAGS_DECIMAL_TS, false, 0 },
321 { "hex", sizeof("hex" ) - 1, RTLOGFLAGS_DECIMAL_TS, true, 0 },
322 { "writethru", sizeof("writethru" ) - 1, RTLOGFLAGS_WRITE_THROUGH, false, 0 },
323 { "writethrough", sizeof("writethrough") - 1, RTLOGFLAGS_WRITE_THROUGH, false, 0 },
324 { "flush", sizeof("flush" ) - 1, RTLOGFLAGS_FLUSH, false, 0 },
325 { "lockcnts", sizeof("lockcnts" ) - 1, RTLOGFLAGS_PREFIX_LOCK_COUNTS, false, 0 },
326 { "cpuid", sizeof("cpuid" ) - 1, RTLOGFLAGS_PREFIX_CPUID, false, 0 },
327 { "pid", sizeof("pid" ) - 1, RTLOGFLAGS_PREFIX_PID, false, 0 },
328 { "flagno", sizeof("flagno" ) - 1, RTLOGFLAGS_PREFIX_FLAG_NO, false, 0 },
329 { "flag", sizeof("flag" ) - 1, RTLOGFLAGS_PREFIX_FLAG, false, 0 },
330 { "groupno", sizeof("groupno" ) - 1, RTLOGFLAGS_PREFIX_GROUP_NO, false, 0 },
331 { "group", sizeof("group" ) - 1, RTLOGFLAGS_PREFIX_GROUP, false, 0 },
332 { "tid", sizeof("tid" ) - 1, RTLOGFLAGS_PREFIX_TID, false, 0 },
333 { "thread", sizeof("thread" ) - 1, RTLOGFLAGS_PREFIX_THREAD, false, 0 },
334 { "custom", sizeof("custom" ) - 1, RTLOGFLAGS_PREFIX_CUSTOM, false, 0 },
335 { "timeprog", sizeof("timeprog" ) - 1, RTLOGFLAGS_PREFIX_TIME_PROG, false, 0 },
336 { "time", sizeof("time" ) - 1, RTLOGFLAGS_PREFIX_TIME, false, 0 },
337 { "msprog", sizeof("msprog" ) - 1, RTLOGFLAGS_PREFIX_MS_PROG, false, 0 },
338 { "tsc", sizeof("tsc" ) - 1, RTLOGFLAGS_PREFIX_TSC, false, 0 }, /* before ts! */
339 { "ts", sizeof("ts" ) - 1, RTLOGFLAGS_PREFIX_TS, false, 0 },
340 /* We intentionally omit RTLOGFLAGS_RESTRICT_GROUPS. */
341};
342
343/**
344 * Logger destination instructions.
345 */
346static struct
347{
348 const char *pszInstr; /**< The name. */
349 size_t cchInstr; /**< The size of the name. */
350 uint32_t fFlag; /**< The corresponding destination flag. */
351} const g_aLogDst[] =
352{
353 { RT_STR_TUPLE("file"), RTLOGDEST_FILE }, /* Must be 1st! */
354 { RT_STR_TUPLE("dir"), RTLOGDEST_FILE }, /* Must be 2nd! */
355 { RT_STR_TUPLE("history"), 0 }, /* Must be 3rd! */
356 { RT_STR_TUPLE("histsize"), 0 }, /* Must be 4th! */
357 { RT_STR_TUPLE("histtime"), 0 }, /* Must be 5th! */
358 { RT_STR_TUPLE("ringbuf"), RTLOGDEST_RINGBUF }, /* Must be 6th! */
359 { RT_STR_TUPLE("stdout"), RTLOGDEST_STDOUT },
360 { RT_STR_TUPLE("stderr"), RTLOGDEST_STDERR },
361 { RT_STR_TUPLE("debugger"), RTLOGDEST_DEBUGGER },
362 { RT_STR_TUPLE("com"), RTLOGDEST_COM },
363 { RT_STR_TUPLE("nodeny"), RTLOGDEST_F_NO_DENY },
364 { RT_STR_TUPLE("user"), RTLOGDEST_USER },
365 /* The RTLOGDEST_FIXED_XXX flags are omitted on purpose. */
366};
367
368#ifdef IN_RING3
369/** Log rotation backoff table - millisecond sleep intervals.
370 * Important on Windows host, especially for VBoxSVC release logging. Only a
371 * medium term solution, until a proper fix for log file handling is available.
372 * 10 seconds total.
373 */
374static const uint32_t g_acMsLogBackoff[] =
375{ 10, 10, 10, 20, 50, 100, 200, 200, 200, 200, 500, 500, 500, 500, 1000, 1000, 1000, 1000, 1000, 1000, 1000 };
376#endif
377
378
379/**
380 * Locks the logger instance.
381 *
382 * @returns See RTSemSpinMutexRequest().
383 * @param pLogger The logger instance.
384 */
385DECLINLINE(int) rtlogLock(PRTLOGGER pLogger)
386{
387#ifndef IN_RC
388 PRTLOGGERINTERNAL pInt = pLogger->pInt;
389 AssertMsgReturn(pInt->uRevision == RTLOGGERINTERNAL_REV, ("%#x != %#x\n", pInt->uRevision, RTLOGGERINTERNAL_REV),
390 VERR_LOG_REVISION_MISMATCH);
391 AssertMsgReturn(pInt->cbSelf == sizeof(*pInt), ("%#x != %#x\n", pInt->cbSelf, sizeof(*pInt)),
392 VERR_LOG_REVISION_MISMATCH);
393 if (pInt->hSpinMtx != NIL_RTSEMSPINMUTEX)
394 {
395 int rc = RTSemSpinMutexRequest(pInt->hSpinMtx);
396 if (RT_FAILURE(rc))
397 return rc;
398 }
399#else
400 NOREF(pLogger);
401#endif
402 return VINF_SUCCESS;
403}
404
405
406/**
407 * Unlocks the logger instance.
408 * @param pLogger The logger instance.
409 */
410DECLINLINE(void) rtlogUnlock(PRTLOGGER pLogger)
411{
412#ifndef IN_RC
413 if (pLogger->pInt->hSpinMtx != NIL_RTSEMSPINMUTEX)
414 RTSemSpinMutexRelease(pLogger->pInt->hSpinMtx);
415#else
416 NOREF(pLogger);
417#endif
418 return;
419}
420
421#ifndef IN_RC
422# ifdef IN_RING3
423
424# ifdef SOME_UNUSED_FUNCTION
425/**
426 * Logging to file, output callback.
427 *
428 * @param pvArg User argument.
429 * @param pachChars Pointer to an array of utf-8 characters.
430 * @param cbChars Number of bytes in the character array pointed to by pachChars.
431 */
432static DECLCALLBACK(size_t) rtlogPhaseWrite(void *pvArg, const char *pachChars, size_t cbChars)
433{
434 PRTLOGGER pLogger = (PRTLOGGER)pvArg;
435 RTFileWrite(pLogger->pInt->hFile, pachChars, cbChars, NULL);
436 return cbChars;
437}
438
439
440/**
441 * Callback to format VBox formatting extentions.
442 * See @ref pg_rt_str_format for a reference on the format types.
443 *
444 * @returns The number of bytes formatted.
445 * @param pvArg Formatter argument.
446 * @param pfnOutput Pointer to output function.
447 * @param pvArgOutput Argument for the output function.
448 * @param ppszFormat Pointer to the format string pointer. Advance this till the char
449 * after the format specifier.
450 * @param pArgs Pointer to the argument list. Use this to fetch the arguments.
451 * @param cchWidth Format Width. -1 if not specified.
452 * @param cchPrecision Format Precision. -1 if not specified.
453 * @param fFlags Flags (RTSTR_NTFS_*).
454 * @param chArgSize The argument size specifier, 'l' or 'L'.
455 */
456static DECLCALLBACK(size_t) rtlogPhaseFormatStr(void *pvArg, PFNRTSTROUTPUT pfnOutput, void *pvArgOutput,
457 const char **ppszFormat, va_list *pArgs, int cchWidth,
458 int cchPrecision, unsigned fFlags, char chArgSize)
459{
460 char ch = *(*ppszFormat)++;
461
462 AssertMsgFailed(("Invalid logger phase format type '%%%c%.10s'!\n", ch, *ppszFormat)); NOREF(ch);
463
464 return 0;
465}
466
467# endif /* SOME_UNUSED_FUNCTION */
468
469
470/**
471 * Log phase callback function, assumes the lock is already held
472 *
473 * @param pLogger The logger instance.
474 * @param pszFormat Format string.
475 * @param ... Optional arguments as specified in the format string.
476 */
477static DECLCALLBACK(void) rtlogPhaseMsgLocked(PRTLOGGER pLogger, const char *pszFormat, ...)
478{
479 va_list args;
480 AssertPtrReturnVoid(pLogger);
481 AssertPtrReturnVoid(pLogger->pInt);
482 Assert(pLogger->pInt->hSpinMtx != NIL_RTSEMSPINMUTEX);
483
484 va_start(args, pszFormat);
485 rtlogLoggerExVLocked(pLogger, 0, ~0U, pszFormat, args);
486 va_end(args);
487}
488
489
490/**
491 * Log phase callback function, assumes the lock is not held.
492 *
493 * @param pLogger The logger instance.
494 * @param pszFormat Format string.
495 * @param ... Optional arguments as specified in the format string.
496 */
497static DECLCALLBACK(void) rtlogPhaseMsgNormal(PRTLOGGER pLogger, const char *pszFormat, ...)
498{
499 va_list args;
500 AssertPtrReturnVoid(pLogger);
501 AssertPtrReturnVoid(pLogger->pInt);
502 Assert(pLogger->pInt->hSpinMtx != NIL_RTSEMSPINMUTEX);
503
504 va_start(args, pszFormat);
505 RTLogLoggerExV(pLogger, 0, ~0U, pszFormat, args);
506 va_end(args);
507}
508
509# endif /* IN_RING3 */
510
511/**
512 * Adjusts the ring buffer.
513 *
514 * @returns IPRT status code.
515 * @param pLogger The logger instance.
516 * @param cbNewSize The new ring buffer size (0 == default).
517 * @param fForce Whether to do this even if the logger instance hasn't
518 * really been fully created yet (i.e. during RTLogCreate).
519 */
520static int rtLogRingBufAdjust(PRTLOGGER pLogger, uint32_t cbNewSize, bool fForce)
521{
522 /*
523 * If this is early logger init, don't do anything.
524 */
525 if (!pLogger->pInt->fCreated && !fForce)
526 return VINF_SUCCESS;
527
528 /*
529 * Lock the logger and make the necessary changes.
530 */
531 int rc = rtlogLock(pLogger);
532 if (RT_SUCCESS(rc))
533 {
534 if (cbNewSize == 0)
535 cbNewSize = RTLOG_RINGBUF_DEFAULT_SIZE;
536 if ( pLogger->pInt->cbRingBuf != cbNewSize
537 || !pLogger->pInt->pchRingBufCur)
538 {
539 uintptr_t offOld = pLogger->pInt->pchRingBufCur - pLogger->pInt->pszRingBuf;
540 if (offOld < sizeof(RTLOG_RINGBUF_EYE_CATCHER))
541 offOld = sizeof(RTLOG_RINGBUF_EYE_CATCHER);
542 else if (offOld >= cbNewSize)
543 {
544 memmove(pLogger->pInt->pszRingBuf, &pLogger->pInt->pszRingBuf[offOld - cbNewSize], cbNewSize);
545 offOld = sizeof(RTLOG_RINGBUF_EYE_CATCHER);
546 }
547
548 void *pvNew = RTMemRealloc(pLogger->pInt->pchRingBufCur, cbNewSize);
549 if (pvNew)
550 {
551 pLogger->pInt->pszRingBuf = (char *)pvNew;
552 pLogger->pInt->pchRingBufCur = (char *)pvNew + offOld;
553 pLogger->pInt->cbRingBuf = cbNewSize;
554 memcpy(pvNew, RTLOG_RINGBUF_EYE_CATCHER, sizeof(RTLOG_RINGBUF_EYE_CATCHER));
555 memcpy((char *)pvNew + cbNewSize - sizeof(RTLOG_RINGBUF_EYE_CATCHER_END),
556 RTLOG_RINGBUF_EYE_CATCHER_END, sizeof(RTLOG_RINGBUF_EYE_CATCHER_END));
557 rc = VINF_SUCCESS;
558 }
559 else
560 rc = VERR_NO_MEMORY;
561 }
562 rtlogUnlock(pLogger);
563 }
564
565 return rc;
566}
567
568
569/**
570 * Writes text to the ring buffer.
571 *
572 * @param pInt The internal logger data structure.
573 * @param pachText The text to write.
574 * @param cchText The number of chars (bytes) to write.
575 */
576static void rtLogRingBufWrite(PRTLOGGERINTERNAL pInt, const char *pachText, size_t cchText)
577{
578 /*
579 * Get the ring buffer data, adjusting it to only describe the writable
580 * part of the buffer.
581 */
582 char * const pchStart = &pInt->pszRingBuf[sizeof(RTLOG_RINGBUF_EYE_CATCHER)];
583 size_t const cchBuf = pInt->cbRingBuf - sizeof(RTLOG_RINGBUF_EYE_CATCHER) - sizeof(RTLOG_RINGBUF_EYE_CATCHER_END);
584 char *pchCur = pInt->pchRingBufCur;
585 size_t cchLeft = pchCur - pchStart;
586 if (RT_LIKELY(cchLeft < cchBuf))
587 cchLeft = cchBuf - cchLeft;
588 else
589 {
590 /* May happen in ring-0 where a thread or two went ahead without getting the lock. */
591 pchCur = pchStart;
592 cchLeft = cchBuf;
593 }
594 Assert(cchBuf < pInt->cbRingBuf);
595
596 if (cchText < cchLeft)
597 {
598 /*
599 * The text fits in the remaining space.
600 */
601 memcpy(pchCur, pachText, cchText);
602 pchCur[cchText] = '\0';
603 pInt->pchRingBufCur = &pchCur[cchText];
604 pInt->cbRingBufUnflushed += cchText;
605 }
606 else
607 {
608 /*
609 * The text wraps around. Taking the simple but inefficient approach
610 * to input texts that are longer than the ring buffer since that
611 * is unlikely to the be a frequent case.
612 */
613 /* Fill to the end of the buffer. */
614 memcpy(pchCur, pachText, cchLeft);
615 pachText += cchLeft;
616 cchText -= cchLeft;
617 pInt->cbRingBufUnflushed += cchLeft;
618 pInt->pchRingBufCur = pchStart;
619
620 /* Ring buffer overflows (the plainly inefficient bit). */
621 while (cchText >= cchBuf)
622 {
623 memcpy(pchStart, pachText, cchBuf);
624 pachText += cchBuf;
625 cchText -= cchBuf;
626 pInt->cbRingBufUnflushed += cchBuf;
627 }
628
629 /* The final bit, if any. */
630 if (cchText > 0)
631 {
632 memcpy(pchStart, pachText, cchText);
633 pInt->cbRingBufUnflushed += cchText;
634 }
635 pchStart[cchText] = '\0';
636 pInt->pchRingBufCur = &pchStart[cchText];
637 }
638}
639
640
641/**
642 * Flushes the ring buffer to all the other log destinations.
643 *
644 * @param pLogger The logger instance which ring buffer should be flushed.
645 */
646static void rtLogRingBufFlush(PRTLOGGER pLogger)
647{
648 const char *pszPreamble;
649 size_t cchPreamble;
650 const char *pszFirst;
651 size_t cchFirst;
652 const char *pszSecond;
653 size_t cchSecond;
654
655 /*
656 * Get the ring buffer data, adjusting it to only describe the writable
657 * part of the buffer.
658 */
659 uint64_t cchUnflushed = pLogger->pInt->cbRingBufUnflushed;
660 char * const pszBuf = &pLogger->pInt->pszRingBuf[sizeof(RTLOG_RINGBUF_EYE_CATCHER)];
661 size_t const cchBuf = pLogger->pInt->cbRingBuf - sizeof(RTLOG_RINGBUF_EYE_CATCHER) - sizeof(RTLOG_RINGBUF_EYE_CATCHER_END);
662 size_t offCur = pLogger->pInt->pchRingBufCur - pszBuf;
663 size_t cchAfter;
664 if (RT_LIKELY(offCur < cchBuf))
665 cchAfter = cchBuf - offCur;
666 else /* May happen in ring-0 where a thread or two went ahead without getting the lock. */
667 {
668 offCur = 0;
669 cchAfter = cchBuf;
670 }
671
672 pLogger->pInt->cbRingBufUnflushed = 0;
673
674 /*
675 * Figure out whether there are one or two segments that needs writing,
676 * making the last segment is terminated. (The first is always
677 * terminated because of the eye-catcher at the end of the buffer.)
678 */
679 if (cchUnflushed == 0)
680 return;
681 pszBuf[offCur] = '\0';
682 if (cchUnflushed >= cchBuf)
683 {
684 pszFirst = &pszBuf[offCur + 1];
685 cchFirst = cchAfter ? cchAfter - 1 : 0;
686 pszSecond = pszBuf;
687 cchSecond = offCur;
688 pszPreamble = "\n*FLUSH RING BUF*\n";
689 cchPreamble = sizeof("\n*FLUSH RING BUF*\n") - 1;
690 }
691 else if ((size_t)cchUnflushed <= offCur)
692 {
693 cchFirst = (size_t)cchUnflushed;
694 pszFirst = &pszBuf[offCur - cchFirst];
695 pszSecond = "";
696 cchSecond = 0;
697 pszPreamble = "";
698 cchPreamble = 0;
699 }
700 else
701 {
702 cchFirst = (size_t)cchUnflushed - offCur;
703 pszFirst = &pszBuf[cchBuf - cchFirst];
704 pszSecond = pszBuf;
705 cchSecond = offCur;
706 pszPreamble = "";
707 cchPreamble = 0;
708 }
709
710 /*
711 * Write the ring buffer to all other destiations.
712 */
713 if (pLogger->fDestFlags & RTLOGDEST_USER)
714 {
715 if (cchPreamble)
716 RTLogWriteUser(pszPreamble, cchPreamble);
717 if (cchFirst)
718 RTLogWriteUser(pszFirst, cchFirst);
719 if (cchSecond)
720 RTLogWriteUser(pszSecond, cchSecond);
721 }
722
723 if (pLogger->fDestFlags & RTLOGDEST_DEBUGGER)
724 {
725 if (cchPreamble)
726 RTLogWriteDebugger(pszPreamble, cchPreamble);
727 if (cchFirst)
728 RTLogWriteDebugger(pszFirst, cchFirst);
729 if (cchSecond)
730 RTLogWriteDebugger(pszSecond, cchSecond);
731 }
732
733# ifdef IN_RING3
734 if (pLogger->fDestFlags & RTLOGDEST_FILE)
735 {
736 if (pLogger->pInt->hFile != NIL_RTFILE)
737 {
738 if (cchPreamble)
739 RTFileWrite(pLogger->pInt->hFile, pszPreamble, cchPreamble, NULL);
740 if (cchFirst)
741 RTFileWrite(pLogger->pInt->hFile, pszFirst, cchFirst, NULL);
742 if (cchSecond)
743 RTFileWrite(pLogger->pInt->hFile, pszSecond, cchSecond, NULL);
744 if (pLogger->fFlags & RTLOGFLAGS_FLUSH)
745 RTFileFlush(pLogger->pInt->hFile);
746 }
747 if (pLogger->pInt->cHistory)
748 pLogger->pInt->cbHistoryFileWritten += cchFirst + cchSecond;
749 }
750# endif
751
752 if (pLogger->fDestFlags & RTLOGDEST_STDOUT)
753 {
754 if (cchPreamble)
755 RTLogWriteStdOut(pszPreamble, cchPreamble);
756 if (cchFirst)
757 RTLogWriteStdOut(pszFirst, cchFirst);
758 if (cchSecond)
759 RTLogWriteStdOut(pszSecond, cchSecond);
760 }
761
762 if (pLogger->fDestFlags & RTLOGDEST_STDERR)
763 {
764 if (cchPreamble)
765 RTLogWriteStdErr(pszPreamble, cchPreamble);
766 if (cchFirst)
767 RTLogWriteStdErr(pszFirst, cchFirst);
768 if (cchSecond)
769 RTLogWriteStdErr(pszSecond, cchSecond);
770 }
771
772# if defined(IN_RING0) && !defined(LOG_NO_COM)
773 if (pLogger->fDestFlags & RTLOGDEST_COM)
774 {
775 if (cchPreamble)
776 RTLogWriteCom(pszPreamble, cchPreamble);
777 if (cchFirst)
778 RTLogWriteCom(pszFirst, cchFirst);
779 if (cchSecond)
780 RTLogWriteCom(pszSecond, cchSecond);
781 }
782# endif
783}
784
785
786RTDECL(int) RTLogCreateExV(PRTLOGGER *ppLogger, uint32_t fFlags, const char *pszGroupSettings, const char *pszEnvVarBase,
787 unsigned cGroups, const char * const *papszGroups, uint32_t cMaxEntriesPerGroup,
788 uint32_t fDestFlags, PFNRTLOGPHASE pfnPhase, uint32_t cHistory,
789 uint64_t cbHistoryFileMax, uint32_t cSecsHistoryTimeSlot,
790 PRTERRINFO pErrInfo, const char *pszFilenameFmt, va_list args)
791{
792 int rc;
793 size_t offInternal;
794 size_t cbLogger;
795 PRTLOGGER pLogger;
796
797 /*
798 * Validate input.
799 */
800 if ( (cGroups && !papszGroups)
801 || !VALID_PTR(ppLogger) )
802 {
803 AssertMsgFailed(("Invalid parameters!\n"));
804 return VERR_INVALID_PARAMETER;
805 }
806 *ppLogger = NULL;
807
808 AssertMsgReturn(cHistory < _1M, ("%#x", cHistory), VERR_OUT_OF_RANGE);
809
810 /*
811 * Allocate a logger instance.
812 */
813 offInternal = RT_UOFFSETOF_DYN(RTLOGGER, afGroups[cGroups]);
814 offInternal = RT_ALIGN_Z(offInternal, sizeof(uint64_t));
815 cbLogger = offInternal + sizeof(RTLOGGERINTERNAL);
816 if (fFlags & RTLOGFLAGS_RESTRICT_GROUPS)
817 cbLogger += cGroups * sizeof(uint32_t);
818 pLogger = (PRTLOGGER)RTMemAllocZVarTag(cbLogger, "may-leak:log-instance");
819 if (pLogger)
820 {
821# if defined(RT_ARCH_X86) && (!defined(LOG_USE_C99) || !defined(RT_WITHOUT_EXEC_ALLOC))
822 uint8_t *pu8Code;
823# endif
824 pLogger->u32Magic = RTLOGGER_MAGIC;
825 pLogger->cGroups = cGroups;
826 pLogger->fFlags = fFlags;
827 pLogger->fDestFlags = fDestFlags;
828 pLogger->pInt = (PRTLOGGERINTERNAL)((uintptr_t)pLogger + offInternal);
829 pLogger->pInt->uRevision = RTLOGGERINTERNAL_REV;
830 pLogger->pInt->cbSelf = sizeof(RTLOGGERINTERNAL);
831 pLogger->pInt->hSpinMtx = NIL_RTSEMSPINMUTEX;
832 pLogger->pInt->pfnFlush = NULL;
833 pLogger->pInt->pfnPrefix = NULL;
834 pLogger->pInt->pvPrefixUserArg = NULL;
835 pLogger->pInt->fPendingPrefix = true;
836 pLogger->pInt->fCreated = false;
837 pLogger->pInt->nsR0ProgramStart = 0;
838 RT_ZERO(pLogger->pInt->szR0ThreadName);
839 pLogger->pInt->cMaxGroups = cGroups;
840 pLogger->pInt->papszGroups = papszGroups;
841 if (fFlags & RTLOGFLAGS_RESTRICT_GROUPS)
842 pLogger->pInt->pacEntriesPerGroup = (uint32_t *)(pLogger->pInt + 1);
843 else
844 pLogger->pInt->pacEntriesPerGroup = NULL;
845 pLogger->pInt->cMaxEntriesPerGroup = cMaxEntriesPerGroup ? cMaxEntriesPerGroup : UINT32_MAX;
846# ifdef IN_RING3
847 pLogger->pInt->pfnPhase = pfnPhase;
848 pLogger->pInt->hFile = NIL_RTFILE;
849 pLogger->pInt->cHistory = cHistory;
850 if (cbHistoryFileMax == 0)
851 pLogger->pInt->cbHistoryFileMax = UINT64_MAX;
852 else
853 pLogger->pInt->cbHistoryFileMax = cbHistoryFileMax;
854 if (cSecsHistoryTimeSlot == 0)
855 pLogger->pInt->cSecsHistoryTimeSlot = UINT32_MAX;
856 else
857 pLogger->pInt->cSecsHistoryTimeSlot = cSecsHistoryTimeSlot;
858# else /* !IN_RING3 */
859 RT_NOREF_PV(pfnPhase); RT_NOREF_PV(cHistory); RT_NOREF_PV(cbHistoryFileMax); RT_NOREF_PV(cSecsHistoryTimeSlot);
860# endif /* !IN_RING3 */
861 if (pszGroupSettings)
862 RTLogGroupSettings(pLogger, pszGroupSettings);
863
864# if defined(RT_ARCH_X86) && (!defined(LOG_USE_C99) || !defined(RT_WITHOUT_EXEC_ALLOC))
865 /*
866 * Emit wrapper code.
867 */
868 pu8Code = (uint8_t *)RTMemExecAlloc(64);
869 if (pu8Code)
870 {
871 pLogger->pfnLogger = *(PFNRTLOGGER*)&pu8Code;
872 *pu8Code++ = 0x68; /* push imm32 */
873 *(void **)pu8Code = pLogger;
874 pu8Code += sizeof(void *);
875 *pu8Code++ = 0xe8; /* call rel32 */
876 *(uint32_t *)pu8Code = (uintptr_t)RTLogLogger - ((uintptr_t)pu8Code + sizeof(uint32_t));
877 pu8Code += sizeof(uint32_t);
878 *pu8Code++ = 0x8d; /* lea esp, [esp + 4] */
879 *pu8Code++ = 0x64;
880 *pu8Code++ = 0x24;
881 *pu8Code++ = 0x04;
882 *pu8Code++ = 0xc3; /* ret near */
883 AssertMsg((uintptr_t)pu8Code - (uintptr_t)pLogger->pfnLogger <= 64,
884 ("Wrapper assembly is too big! %d bytes\n", (uintptr_t)pu8Code - (uintptr_t)pLogger->pfnLogger));
885 rc = VINF_SUCCESS;
886 }
887 else
888 {
889 rc = VERR_NO_MEMORY;
890# ifdef RT_OS_LINUX
891 /* Most probably SELinux causing trouble since the larger RTMemAlloc succeeded. */
892 RTErrInfoSet(pErrInfo, rc, N_("mmap(PROT_WRITE | PROT_EXEC) failed -- SELinux?"));
893# endif
894 }
895 if (RT_SUCCESS(rc))
896# endif /* X86 wrapper code*/
897 {
898# ifdef IN_RING3 /* files and env.vars. are only accessible when in R3 at the present time. */
899 /*
900 * Format the filename.
901 */
902 if (pszFilenameFmt)
903 {
904 /** @todo validate the length, fail on overflow. */
905 RTStrPrintfV(pLogger->pInt->szFilename, sizeof(pLogger->pInt->szFilename), pszFilenameFmt, args);
906 if (pLogger->pInt->szFilename[0])
907 pLogger->fDestFlags |= RTLOGDEST_FILE;
908 }
909
910 /*
911 * Parse the environment variables.
912 */
913 if (pszEnvVarBase)
914 {
915 /* make temp copy of environment variable base. */
916 size_t cchEnvVarBase = strlen(pszEnvVarBase);
917 char *pszEnvVar = (char *)alloca(cchEnvVarBase + 16);
918 memcpy(pszEnvVar, pszEnvVarBase, cchEnvVarBase);
919
920 /*
921 * Destination.
922 */
923 strcpy(pszEnvVar + cchEnvVarBase, "_DEST");
924 const char *pszValue = RTEnvGet(pszEnvVar);
925 if (pszValue)
926 RTLogDestinations(pLogger, pszValue);
927
928 /*
929 * The flags.
930 */
931 strcpy(pszEnvVar + cchEnvVarBase, "_FLAGS");
932 pszValue = RTEnvGet(pszEnvVar);
933 if (pszValue)
934 RTLogFlags(pLogger, pszValue);
935
936 /*
937 * The group settings.
938 */
939 pszEnvVar[cchEnvVarBase] = '\0';
940 pszValue = RTEnvGet(pszEnvVar);
941 if (pszValue)
942 RTLogGroupSettings(pLogger, pszValue);
943
944 /*
945 * Group limit.
946 */
947 strcpy(pszEnvVar + cchEnvVarBase, "_MAX_PER_GROUP");
948 pszValue = RTEnvGet(pszEnvVar);
949 if (pszValue)
950 {
951 uint32_t cMax;
952 rc = RTStrToUInt32Full(pszValue, 0, &cMax);
953 if (RT_SUCCESS(rc))
954 pLogger->pInt->cMaxEntriesPerGroup = cMax ? cMax : UINT32_MAX;
955 else
956 AssertMsgFailed(("Invalid group limit! %s=%s\n", pszEnvVar, pszValue));
957 }
958
959 }
960# else /* !IN_RING3 */
961 RT_NOREF_PV(pszEnvVarBase); RT_NOREF_PV(pszFilenameFmt); RT_NOREF_PV(args);
962# endif /* !IN_RING3 */
963
964 /*
965 * Open the destination(s).
966 */
967 rc = VINF_SUCCESS;
968 if ((pLogger->fDestFlags & (RTLOGDEST_F_DELAY_FILE | RTLOGDEST_FILE)) == RTLOGDEST_F_DELAY_FILE)
969 pLogger->fDestFlags &= ~RTLOGDEST_F_DELAY_FILE;
970# ifdef IN_RING3
971 if ((pLogger->fDestFlags & (RTLOGDEST_FILE | RTLOGDEST_F_DELAY_FILE)) == RTLOGDEST_FILE)
972 rc = rtR3LogOpenFileDestination(pLogger, pErrInfo);
973# endif
974
975 if ((pLogger->fDestFlags & RTLOGDEST_RINGBUF) && RT_SUCCESS(rc))
976 rc = rtLogRingBufAdjust(pLogger, pLogger->pInt->cbRingBuf, true /*fForce*/);
977
978 /*
979 * Create mutex and check how much it counts when entering the lock
980 * so that we can report the values for RTLOGFLAGS_PREFIX_LOCK_COUNTS.
981 */
982 if (RT_SUCCESS(rc))
983 {
984 rc = RTSemSpinMutexCreate(&pLogger->pInt->hSpinMtx, RTSEMSPINMUTEX_FLAGS_IRQ_SAFE);
985 if (RT_SUCCESS(rc))
986 {
987# ifdef IN_RING3 /** @todo do counters in ring-0 too? */
988 RTTHREAD Thread = RTThreadSelf();
989 if (Thread != NIL_RTTHREAD)
990 {
991 int32_t c = RTLockValidatorWriteLockGetCount(Thread);
992 RTSemSpinMutexRequest(pLogger->pInt->hSpinMtx);
993 c = RTLockValidatorWriteLockGetCount(Thread) - c;
994 RTSemSpinMutexRelease(pLogger->pInt->hSpinMtx);
995 ASMAtomicWriteU32(&g_cLoggerLockCount, c);
996 }
997
998 /* Use the callback to generate some initial log contents. */
999 Assert(VALID_PTR(pLogger->pInt->pfnPhase) || pLogger->pInt->pfnPhase == NULL);
1000 if (pLogger->pInt->pfnPhase)
1001 pLogger->pInt->pfnPhase(pLogger, RTLOGPHASE_BEGIN, rtlogPhaseMsgNormal);
1002# endif
1003 pLogger->pInt->fCreated = true;
1004 *ppLogger = pLogger;
1005 return VINF_SUCCESS;
1006 }
1007
1008 RTErrInfoSet(pErrInfo, rc, N_("failed to create semaphore"));
1009 }
1010# ifdef IN_RING3
1011 RTFileClose(pLogger->pInt->hFile);
1012# endif
1013# if defined(LOG_USE_C99) && defined(RT_WITHOUT_EXEC_ALLOC)
1014 RTMemFree(*(void **)&pLogger->pfnLogger);
1015# else
1016 RTMemExecFree(*(void **)&pLogger->pfnLogger, 64);
1017# endif
1018 }
1019 RTMemFree(pLogger);
1020 }
1021 else
1022 rc = VERR_NO_MEMORY;
1023
1024 return rc;
1025}
1026RT_EXPORT_SYMBOL(RTLogCreateExV);
1027
1028
1029RTDECL(int) RTLogCreate(PRTLOGGER *ppLogger, uint32_t fFlags, const char *pszGroupSettings,
1030 const char *pszEnvVarBase, unsigned cGroups, const char * const * papszGroups,
1031 uint32_t fDestFlags, const char *pszFilenameFmt, ...)
1032{
1033 va_list args;
1034 int rc;
1035
1036 va_start(args, pszFilenameFmt);
1037 rc = RTLogCreateExV(ppLogger, fFlags, pszGroupSettings, pszEnvVarBase,
1038 cGroups, papszGroups, UINT32_MAX /*cMaxEntriesPerGroup*/, fDestFlags,
1039 NULL /*pfnPhase*/, 0 /*cHistory*/, 0 /*cbHistoryFileMax*/, 0 /*cSecsHistoryTimeSlot*/,
1040 NULL /*pErrInfo*/, pszFilenameFmt, args);
1041 va_end(args);
1042 return rc;
1043}
1044RT_EXPORT_SYMBOL(RTLogCreate);
1045
1046
1047RTDECL(int) RTLogCreateEx(PRTLOGGER *ppLogger, uint32_t fFlags, const char *pszGroupSettings, const char *pszEnvVarBase,
1048 unsigned cGroups, const char * const *papszGroups, uint32_t cMaxEntriesPerGroup,
1049 uint32_t fDestFlags, PFNRTLOGPHASE pfnPhase, uint32_t cHistory,
1050 uint64_t cbHistoryFileMax, uint32_t cSecsHistoryTimeSlot,
1051 PRTERRINFO pErrInfo, const char *pszFilenameFmt, ...)
1052{
1053 va_list args;
1054 int rc;
1055
1056 va_start(args, pszFilenameFmt);
1057 rc = RTLogCreateExV(ppLogger, fFlags, pszGroupSettings, pszEnvVarBase, cGroups, papszGroups, cMaxEntriesPerGroup,
1058 fDestFlags, pfnPhase, cHistory, cbHistoryFileMax, cSecsHistoryTimeSlot,
1059 pErrInfo, pszFilenameFmt, args);
1060 va_end(args);
1061 return rc;
1062}
1063RT_EXPORT_SYMBOL(RTLogCreateEx);
1064
1065
1066/**
1067 * Destroys a logger instance.
1068 *
1069 * The instance is flushed and all output destinations closed (where applicable).
1070 *
1071 * @returns iprt status code.
1072 * @param pLogger The logger instance which close destroyed. NULL is fine.
1073 */
1074RTDECL(int) RTLogDestroy(PRTLOGGER pLogger)
1075{
1076 int rc;
1077 uint32_t iGroup;
1078 RTSEMSPINMUTEX hSpinMtx;
1079
1080 /*
1081 * Validate input.
1082 */
1083 if (!pLogger)
1084 return VINF_SUCCESS;
1085 AssertPtrReturn(pLogger, VERR_INVALID_POINTER);
1086 AssertReturn(pLogger->u32Magic == RTLOGGER_MAGIC, VERR_INVALID_MAGIC);
1087 AssertPtrReturn(pLogger->pInt, VERR_INVALID_POINTER);
1088
1089 /*
1090 * Acquire logger instance sem and disable all logging. (paranoia)
1091 */
1092 rc = rtlogLock(pLogger);
1093 AssertMsgRCReturn(rc, ("%Rrc\n", rc), rc);
1094
1095 pLogger->fFlags |= RTLOGFLAGS_DISABLED;
1096 iGroup = pLogger->cGroups;
1097 while (iGroup-- > 0)
1098 pLogger->afGroups[iGroup] = 0;
1099
1100 /*
1101 * Flush it.
1102 */
1103 rtlogFlush(pLogger, false /*fNeedSpace*/);
1104
1105# ifdef IN_RING3
1106 /*
1107 * Add end of logging message.
1108 */
1109 if ( (pLogger->fDestFlags & RTLOGDEST_FILE)
1110 && pLogger->pInt->hFile != NIL_RTFILE)
1111 pLogger->pInt->pfnPhase(pLogger, RTLOGPHASE_END, rtlogPhaseMsgLocked);
1112
1113 /*
1114 * Close output stuffs.
1115 */
1116 if (pLogger->pInt->hFile != NIL_RTFILE)
1117 {
1118 int rc2 = RTFileClose(pLogger->pInt->hFile);
1119 AssertRC(rc2);
1120 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
1121 rc = rc2;
1122 pLogger->pInt->hFile = NIL_RTFILE;
1123 }
1124# endif
1125
1126 /*
1127 * Free the mutex, the wrapper and the instance memory.
1128 */
1129 hSpinMtx = pLogger->pInt->hSpinMtx;
1130 pLogger->pInt->hSpinMtx = NIL_RTSEMSPINMUTEX;
1131 if (hSpinMtx != NIL_RTSEMSPINMUTEX)
1132 {
1133 int rc2;
1134 RTSemSpinMutexRelease(hSpinMtx);
1135 rc2 = RTSemSpinMutexDestroy(hSpinMtx);
1136 AssertRC(rc2);
1137 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
1138 rc = rc2;
1139 }
1140
1141 if (pLogger->pfnLogger)
1142 {
1143# if defined(LOG_USE_C99) && defined(RT_WITHOUT_EXEC_ALLOC)
1144 RTMemFree(*(void **)&pLogger->pfnLogger);
1145# else
1146 RTMemExecFree(*(void **)&pLogger->pfnLogger, 64);
1147# endif
1148 pLogger->pfnLogger = NULL;
1149 }
1150 RTMemFree(pLogger);
1151
1152 return rc;
1153}
1154RT_EXPORT_SYMBOL(RTLogDestroy);
1155
1156
1157/**
1158 * Create a logger instance clone for RC usage.
1159 *
1160 * @returns iprt status code.
1161 *
1162 * @param pLogger The logger instance to be cloned.
1163 * @param pLoggerRC Where to create the RC logger instance.
1164 * @param cbLoggerRC Amount of memory allocated to for the RC logger
1165 * instance clone.
1166 * @param pfnLoggerRCPtr Pointer to logger wrapper function for this
1167 * instance (RC Ptr).
1168 * @param pfnFlushRCPtr Pointer to flush function (RC Ptr).
1169 * @param fFlags Logger instance flags, a combination of the RTLOGFLAGS_* values.
1170 */
1171RTDECL(int) RTLogCloneRC(PRTLOGGER pLogger, PRTLOGGERRC pLoggerRC, size_t cbLoggerRC,
1172 RTRCPTR pfnLoggerRCPtr, RTRCPTR pfnFlushRCPtr, uint32_t fFlags)
1173{
1174 /*
1175 * Validate input.
1176 */
1177 if ( !pLoggerRC
1178 || !pfnFlushRCPtr
1179 || !pfnLoggerRCPtr)
1180 {
1181 AssertMsgFailed(("Invalid parameters!\n"));
1182 return VERR_INVALID_PARAMETER;
1183 }
1184 if (cbLoggerRC < sizeof(*pLoggerRC))
1185 {
1186 AssertMsgFailed(("%d min=%d\n", cbLoggerRC, sizeof(*pLoggerRC)));
1187 return VERR_INVALID_PARAMETER;
1188 }
1189
1190 /*
1191 * Initialize GC instance.
1192 */
1193 pLoggerRC->offScratch = 0;
1194 pLoggerRC->fPendingPrefix = false;
1195 pLoggerRC->pfnLogger = pfnLoggerRCPtr;
1196 pLoggerRC->pfnFlush = pfnFlushRCPtr;
1197 pLoggerRC->u32Magic = RTLOGGERRC_MAGIC;
1198 pLoggerRC->fFlags = fFlags | RTLOGFLAGS_DISABLED;
1199 pLoggerRC->cGroups = 1;
1200 pLoggerRC->afGroups[0] = 0;
1201
1202 /*
1203 * Resolve defaults.
1204 */
1205 if (!pLogger)
1206 {
1207 pLogger = RTLogDefaultInstance();
1208 if (!pLogger)
1209 return VINF_SUCCESS;
1210 }
1211
1212 /*
1213 * Check if there's enough space for the groups.
1214 */
1215 if (cbLoggerRC < (size_t)RT_UOFFSETOF_DYN(RTLOGGERRC, afGroups[pLogger->cGroups]))
1216 {
1217 AssertMsgFailed(("%zu req=%zu cGroups=%d\n", cbLoggerRC, RT_UOFFSETOF_DYN(RTLOGGERRC, afGroups[pLogger->cGroups]), pLogger->cGroups));
1218 return VERR_BUFFER_OVERFLOW;
1219 }
1220 memcpy(&pLoggerRC->afGroups[0], &pLogger->afGroups[0], pLogger->cGroups * sizeof(pLoggerRC->afGroups[0]));
1221 pLoggerRC->cGroups = pLogger->cGroups;
1222
1223 /*
1224 * Copy bits from the HC instance.
1225 */
1226 pLoggerRC->fPendingPrefix = pLogger->pInt->fPendingPrefix;
1227 pLoggerRC->fFlags |= pLogger->fFlags;
1228
1229 /*
1230 * Check if we can remove the disabled flag.
1231 */
1232 if ( pLogger->fDestFlags
1233 && !((pLogger->fFlags | fFlags) & RTLOGFLAGS_DISABLED))
1234 pLoggerRC->fFlags &= ~RTLOGFLAGS_DISABLED;
1235
1236 return VINF_SUCCESS;
1237}
1238RT_EXPORT_SYMBOL(RTLogCloneRC);
1239
1240
1241/**
1242 * Flushes a RC logger instance to a R3 logger.
1243 *
1244 *
1245 * @returns iprt status code.
1246 * @param pLogger The R3 logger instance to flush pLoggerRC to. If NULL
1247 * the default logger is used.
1248 * @param pLoggerRC The RC logger instance to flush.
1249 */
1250RTDECL(void) RTLogFlushRC(PRTLOGGER pLogger, PRTLOGGERRC pLoggerRC)
1251{
1252 /*
1253 * Resolve defaults.
1254 */
1255 if (!pLogger)
1256 {
1257 pLogger = RTLogDefaultInstance();
1258 if (!pLogger)
1259 {
1260 pLoggerRC->offScratch = 0;
1261 return;
1262 }
1263 }
1264
1265 /*
1266 * Any thing to flush?
1267 */
1268 if ( pLogger->offScratch
1269 || pLoggerRC->offScratch)
1270 {
1271 /*
1272 * Acquire logger instance sem.
1273 */
1274 int rc = rtlogLock(pLogger);
1275 if (RT_FAILURE(rc))
1276 return;
1277
1278 /*
1279 * Write whatever the GC instance contains to the HC one, and then
1280 * flush the HC instance.
1281 */
1282 if (pLoggerRC->offScratch)
1283 {
1284 rtLogOutput(pLogger, pLoggerRC->achScratch, pLoggerRC->offScratch);
1285 rtLogOutput(pLogger, NULL, 0);
1286 pLoggerRC->offScratch = 0;
1287 }
1288
1289 /*
1290 * Release the semaphore.
1291 */
1292 rtlogUnlock(pLogger);
1293 }
1294}
1295RT_EXPORT_SYMBOL(RTLogFlushRC);
1296
1297# ifdef IN_RING3
1298
1299RTDECL(int) RTLogCreateForR0(PRTLOGGER pLogger, size_t cbLogger,
1300 RTR0PTR pLoggerR0Ptr, RTR0PTR pfnLoggerR0Ptr, RTR0PTR pfnFlushR0Ptr,
1301 uint32_t fFlags, uint32_t fDestFlags, char const *pszThreadName)
1302{
1303 /*
1304 * Validate input.
1305 */
1306 AssertPtrReturn(pLogger, VERR_INVALID_PARAMETER);
1307 size_t const cbRequired = sizeof(*pLogger) + RTLOGGERINTERNAL_R0_SIZE;
1308 AssertReturn(cbLogger >= cbRequired, VERR_BUFFER_OVERFLOW);
1309 AssertReturn(pLoggerR0Ptr != NIL_RTR0PTR, VERR_INVALID_PARAMETER);
1310 AssertReturn(pfnLoggerR0Ptr != NIL_RTR0PTR, VERR_INVALID_PARAMETER);
1311 size_t const cchThreadName = pszThreadName ? strlen(pszThreadName) : 0;
1312 AssertReturn(cchThreadName < sizeof(pLogger->pInt->szR0ThreadName), VERR_INVALID_NAME);
1313
1314 /*
1315 * Initialize the ring-0 instance.
1316 */
1317 pLogger->achScratch[0] = 0;
1318 pLogger->offScratch = 0;
1319 pLogger->pfnLogger = (PFNRTLOGGER)pfnLoggerR0Ptr;
1320 pLogger->fFlags = fFlags;
1321 pLogger->fDestFlags = fDestFlags & ~RTLOGDEST_FILE;
1322 pLogger->pInt = NULL;
1323 pLogger->cGroups = 1;
1324 pLogger->afGroups[0] = 0;
1325
1326 uint32_t cMaxGroups = (uint32_t)((cbLogger - cbRequired) / sizeof(pLogger->afGroups[0]));
1327 if (fFlags & RTLOGFLAGS_RESTRICT_GROUPS)
1328 cMaxGroups /= 2;
1329 PRTLOGGERINTERNAL pInt;
1330 for (;;)
1331 {
1332 AssertReturn(cMaxGroups > 0, VERR_BUFFER_OVERFLOW);
1333 pInt = (PRTLOGGERINTERNAL)&pLogger->afGroups[cMaxGroups];
1334 if (!((uintptr_t)pInt & (sizeof(uint64_t) - 1)))
1335 break;
1336 cMaxGroups--;
1337 }
1338 pLogger->pInt = (PRTLOGGERINTERNAL)(pLoggerR0Ptr + (uintptr_t)pInt - (uintptr_t)pLogger);
1339 pInt->uRevision = RTLOGGERINTERNAL_REV;
1340 pInt->cbSelf = RTLOGGERINTERNAL_R0_SIZE;
1341 pInt->hSpinMtx = NIL_RTSEMSPINMUTEX; /* Not serialized. */
1342 pInt->pfnFlush = (PFNRTLOGFLUSH)pfnFlushR0Ptr;
1343 pInt->pfnPrefix = NULL;
1344 pInt->pvPrefixUserArg = NULL;
1345 pInt->fPendingPrefix = true;
1346 pInt->cMaxGroups = cMaxGroups;
1347 pInt->papszGroups = NULL;
1348 pInt->cMaxEntriesPerGroup = UINT32_MAX;
1349 if (fFlags & RTLOGFLAGS_RESTRICT_GROUPS)
1350 {
1351 memset(pInt + 1, 0, sizeof(uint32_t) * cMaxGroups);
1352 pInt->pacEntriesPerGroup= (uint32_t *)(pLogger->pInt + 1);
1353 }
1354 else
1355 pInt->pacEntriesPerGroup= NULL;
1356 pInt->nsR0ProgramStart = RTTimeProgramStartNanoTS();
1357 RT_ZERO(pInt->szR0ThreadName);
1358 if (cchThreadName)
1359 memcpy(pInt->szR0ThreadName, pszThreadName, cchThreadName);
1360
1361 pInt->fCreated = true;
1362 pLogger->u32Magic = RTLOGGER_MAGIC;
1363 return VINF_SUCCESS;
1364}
1365RT_EXPORT_SYMBOL(RTLogCreateForR0);
1366
1367
1368RTDECL(size_t) RTLogCalcSizeForR0(uint32_t cGroups, uint32_t fFlags)
1369{
1370 size_t cb = RT_UOFFSETOF_DYN(RTLOGGER, afGroups[cGroups]);
1371 cb = RT_ALIGN_Z(cb, sizeof(uint64_t));
1372 cb += sizeof(RTLOGGERINTERNAL);
1373 if (fFlags & RTLOGFLAGS_RESTRICT_GROUPS)
1374 cb += sizeof(uint32_t) * cGroups;
1375 return cb;
1376}
1377RT_EXPORT_SYMBOL(RTLogCalcSizeForR0);
1378
1379
1380RTDECL(int) RTLogCopyGroupsAndFlagsForR0(PRTLOGGER pDstLogger, RTR0PTR pDstLoggerR0Ptr,
1381 PCRTLOGGER pSrcLogger, uint32_t fFlagsOr, uint32_t fFlagsAnd)
1382{
1383 /*
1384 * Validate input.
1385 */
1386 AssertPtrReturn(pDstLogger, VERR_INVALID_PARAMETER);
1387 AssertPtrNullReturn(pSrcLogger, VERR_INVALID_PARAMETER);
1388
1389 /*
1390 * Resolve defaults.
1391 */
1392 if (!pSrcLogger)
1393 {
1394 pSrcLogger = RTLogDefaultInstance();
1395 if (!pSrcLogger)
1396 {
1397 pDstLogger->fFlags |= RTLOGFLAGS_DISABLED | fFlagsOr;
1398 pDstLogger->cGroups = 1;
1399 pDstLogger->afGroups[0] = 0;
1400 return VINF_SUCCESS;
1401 }
1402 }
1403
1404 /*
1405 * Copy flags and group settings.
1406 */
1407 pDstLogger->fFlags = (pSrcLogger->fFlags & fFlagsAnd & ~RTLOGFLAGS_RESTRICT_GROUPS) | fFlagsOr;
1408
1409 PRTLOGGERINTERNAL pDstInt = (PRTLOGGERINTERNAL)((uintptr_t)pDstLogger->pInt - pDstLoggerR0Ptr + (uintptr_t)pDstLogger);
1410 int rc = VINF_SUCCESS;
1411 uint32_t cGroups = pSrcLogger->cGroups;
1412 if (cGroups > pDstInt->cMaxGroups)
1413 {
1414 AssertMsgFailed(("cMaxGroups=%zd cGroups=%zd (min size %d)\n", pDstInt->cMaxGroups,
1415 pSrcLogger->cGroups, RT_UOFFSETOF_DYN(RTLOGGER, afGroups[pSrcLogger->cGroups]) + RTLOGGERINTERNAL_R0_SIZE));
1416 rc = VERR_INVALID_PARAMETER;
1417 cGroups = pDstInt->cMaxGroups;
1418 }
1419 memcpy(&pDstLogger->afGroups[0], &pSrcLogger->afGroups[0], cGroups * sizeof(pDstLogger->afGroups[0]));
1420 pDstLogger->cGroups = cGroups;
1421
1422 return rc;
1423}
1424RT_EXPORT_SYMBOL(RTLogCopyGroupsAndFlagsForR0);
1425
1426
1427RTDECL(void) RTLogFlushR0(PRTLOGGER pLogger, PRTLOGGER pLoggerR0)
1428{
1429 /*
1430 * Resolve defaults.
1431 */
1432 if (!pLogger)
1433 {
1434 pLogger = RTLogDefaultInstance();
1435 if (!pLogger)
1436 {
1437 /* flushing to "/dev/null". */
1438 if (pLoggerR0->offScratch)
1439 pLoggerR0->offScratch = 0;
1440 return;
1441 }
1442 }
1443
1444 /*
1445 * Anything to flush?
1446 */
1447 if ( pLoggerR0->offScratch
1448 || pLogger->offScratch)
1449 {
1450 /*
1451 * Acquire logger semaphores.
1452 */
1453 int rc = rtlogLock(pLogger);
1454 if (RT_FAILURE(rc))
1455 return;
1456 if (RT_SUCCESS(rc))
1457 {
1458 /*
1459 * Write whatever the GC instance contains to the HC one, and then
1460 * flush the HC instance.
1461 */
1462 if (pLoggerR0->offScratch)
1463 {
1464 rtLogOutput(pLogger, pLoggerR0->achScratch, pLoggerR0->offScratch);
1465 rtLogOutput(pLogger, NULL, 0);
1466 pLoggerR0->offScratch = 0;
1467 }
1468 }
1469 rtlogUnlock(pLogger);
1470 }
1471}
1472RT_EXPORT_SYMBOL(RTLogFlushR0);
1473
1474# endif /* IN_RING3 */
1475
1476
1477/**
1478 * Flushes the buffer in one logger instance onto another logger.
1479 *
1480 * @returns iprt status code.
1481 *
1482 * @param pSrcLogger The logger instance to flush.
1483 * @param pDstLogger The logger instance to flush onto.
1484 * If NULL the default logger will be used.
1485 */
1486RTDECL(void) RTLogFlushToLogger(PRTLOGGER pSrcLogger, PRTLOGGER pDstLogger)
1487{
1488 /*
1489 * Resolve defaults.
1490 */
1491 if (!pDstLogger)
1492 {
1493 pDstLogger = RTLogDefaultInstance();
1494 if (!pDstLogger)
1495 {
1496 /* flushing to "/dev/null". */
1497 if (pSrcLogger->offScratch)
1498 {
1499 int rc = rtlogLock(pSrcLogger);
1500 if (RT_SUCCESS(rc))
1501 {
1502 pSrcLogger->offScratch = 0;
1503 rtlogUnlock(pSrcLogger);
1504 }
1505 }
1506 return;
1507 }
1508 }
1509
1510 /*
1511 * Any thing to flush?
1512 */
1513 if ( pSrcLogger->offScratch
1514 || pDstLogger->offScratch)
1515 {
1516 /*
1517 * Acquire logger semaphores.
1518 */
1519 int rc = rtlogLock(pDstLogger);
1520 if (RT_FAILURE(rc))
1521 return;
1522 rc = rtlogLock(pSrcLogger);
1523 if (RT_SUCCESS(rc))
1524 {
1525 /*
1526 * Write whatever the GC instance contains to the HC one, and then
1527 * flush the HC instance.
1528 */
1529 if (pSrcLogger->offScratch)
1530 {
1531 rtLogOutput(pDstLogger, pSrcLogger->achScratch, pSrcLogger->offScratch);
1532 rtLogOutput(pDstLogger, NULL, 0);
1533 pSrcLogger->offScratch = 0;
1534 }
1535
1536 /*
1537 * Release the semaphores.
1538 */
1539 rtlogUnlock(pSrcLogger);
1540 }
1541 rtlogUnlock(pDstLogger);
1542 }
1543}
1544RT_EXPORT_SYMBOL(RTLogFlushToLogger);
1545
1546
1547/**
1548 * Sets the custom prefix callback.
1549 *
1550 * @returns IPRT status code.
1551 * @param pLogger The logger instance.
1552 * @param pfnCallback The callback.
1553 * @param pvUser The user argument for the callback.
1554 * */
1555RTDECL(int) RTLogSetCustomPrefixCallback(PRTLOGGER pLogger, PFNRTLOGPREFIX pfnCallback, void *pvUser)
1556{
1557 /*
1558 * Resolve defaults.
1559 */
1560 if (!pLogger)
1561 {
1562 pLogger = RTLogDefaultInstance();
1563 if (!pLogger)
1564 return VINF_SUCCESS;
1565 }
1566 AssertReturn(pLogger->u32Magic == RTLOGGER_MAGIC, VERR_INVALID_MAGIC);
1567
1568 /*
1569 * Do the work.
1570 */
1571 rtlogLock(pLogger);
1572 pLogger->pInt->pvPrefixUserArg = pvUser;
1573 pLogger->pInt->pfnPrefix = pfnCallback;
1574 rtlogUnlock(pLogger);
1575
1576 return VINF_SUCCESS;
1577}
1578RT_EXPORT_SYMBOL(RTLogSetCustomPrefixCallback);
1579
1580
1581/**
1582 * Matches a group name with a pattern mask in an case insensitive manner (ASCII).
1583 *
1584 * @returns true if matching and *ppachMask set to the end of the pattern.
1585 * @returns false if no match.
1586 * @param pszGrp The group name.
1587 * @param ppachMask Pointer to the pointer to the mask. Only wildcard supported is '*'.
1588 * @param cchMask The length of the mask, including modifiers. The modifiers is why
1589 * we update *ppachMask on match.
1590 */
1591static bool rtlogIsGroupMatching(const char *pszGrp, const char **ppachMask, size_t cchMask)
1592{
1593 const char *pachMask;
1594
1595 if (!pszGrp || !*pszGrp)
1596 return false;
1597 pachMask = *ppachMask;
1598 for (;;)
1599 {
1600 if (RT_C_TO_LOWER(*pszGrp) != RT_C_TO_LOWER(*pachMask))
1601 {
1602 const char *pszTmp;
1603
1604 /*
1605 * Check for wildcard and do a minimal match if found.
1606 */
1607 if (*pachMask != '*')
1608 return false;
1609
1610 /* eat '*'s. */
1611 do pachMask++;
1612 while (--cchMask && *pachMask == '*');
1613
1614 /* is there more to match? */
1615 if ( !cchMask
1616 || *pachMask == '.'
1617 || *pachMask == '=')
1618 break; /* we're good */
1619
1620 /* do extremely minimal matching (fixme) */
1621 pszTmp = strchr(pszGrp, RT_C_TO_LOWER(*pachMask));
1622 if (!pszTmp)
1623 pszTmp = strchr(pszGrp, RT_C_TO_UPPER(*pachMask));
1624 if (!pszTmp)
1625 return false;
1626 pszGrp = pszTmp;
1627 continue;
1628 }
1629
1630 /* done? */
1631 if (!*++pszGrp)
1632 {
1633 /* trailing wildcard is ok. */
1634 do
1635 {
1636 pachMask++;
1637 cchMask--;
1638 } while (cchMask && *pachMask == '*');
1639 if ( !cchMask
1640 || *pachMask == '.'
1641 || *pachMask == '=')
1642 break; /* we're good */
1643 return false;
1644 }
1645
1646 if (!--cchMask)
1647 return false;
1648 pachMask++;
1649 }
1650
1651 /* match */
1652 *ppachMask = pachMask;
1653 return true;
1654}
1655
1656
1657/**
1658 * Updates the group settings for the logger instance using the specified
1659 * specification string.
1660 *
1661 * @returns iprt status code.
1662 * Failures can safely be ignored.
1663 * @param pLogger Logger instance.
1664 * @param pszValue Value to parse.
1665 */
1666RTDECL(int) RTLogGroupSettings(PRTLOGGER pLogger, const char *pszValue)
1667{
1668 /*
1669 * Resolve defaults.
1670 */
1671 if (!pLogger)
1672 {
1673 pLogger = RTLogDefaultInstance();
1674 if (!pLogger)
1675 return VINF_SUCCESS;
1676 }
1677
1678 /*
1679 * Iterate the string.
1680 */
1681 while (*pszValue)
1682 {
1683 /*
1684 * Skip prefixes (blanks, ;, + and -).
1685 */
1686 bool fEnabled = true;
1687 char ch;
1688 const char *pszStart;
1689 unsigned i;
1690 size_t cch;
1691
1692 while ((ch = *pszValue) == '+' || ch == '-' || ch == ' ' || ch == '\t' || ch == '\n' || ch == ';')
1693 {
1694 if (ch == '+' || ch == '-' || ch == ';')
1695 fEnabled = ch != '-';
1696 pszValue++;
1697 }
1698 if (!*pszValue)
1699 break;
1700
1701 /*
1702 * Find end.
1703 */
1704 pszStart = pszValue;
1705 while ((ch = *pszValue) != '\0' && ch != '+' && ch != '-' && ch != ' ' && ch != '\t')
1706 pszValue++;
1707
1708 /*
1709 * Find the group (ascii case insensitive search).
1710 * Special group 'all'.
1711 */
1712 cch = pszValue - pszStart;
1713 if ( cch >= 3
1714 && (pszStart[0] == 'a' || pszStart[0] == 'A')
1715 && (pszStart[1] == 'l' || pszStart[1] == 'L')
1716 && (pszStart[2] == 'l' || pszStart[2] == 'L')
1717 && (cch == 3 || pszStart[3] == '.' || pszStart[3] == '='))
1718 {
1719 /*
1720 * All.
1721 */
1722 unsigned fFlags = cch == 3
1723 ? RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1
1724 : rtlogGroupFlags(&pszStart[3]);
1725 for (i = 0; i < pLogger->cGroups; i++)
1726 {
1727 if (fEnabled)
1728 pLogger->afGroups[i] |= fFlags;
1729 else
1730 pLogger->afGroups[i] &= ~fFlags;
1731 }
1732 }
1733 else
1734 {
1735 /*
1736 * Specific group(s).
1737 */
1738 for (i = 0; i < pLogger->cGroups; i++)
1739 {
1740 const char *psz2 = (const char*)pszStart;
1741 if (rtlogIsGroupMatching(pLogger->pInt->papszGroups[i], &psz2, cch))
1742 {
1743 unsigned fFlags = RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1;
1744 if (*psz2 == '.' || *psz2 == '=')
1745 fFlags = rtlogGroupFlags(psz2);
1746 if (fEnabled)
1747 pLogger->afGroups[i] |= fFlags;
1748 else
1749 pLogger->afGroups[i] &= ~fFlags;
1750 }
1751 } /* for each group */
1752 }
1753
1754 } /* parse specification */
1755
1756 return VINF_SUCCESS;
1757}
1758RT_EXPORT_SYMBOL(RTLogGroupSettings);
1759
1760
1761/**
1762 * Interprets the group flags suffix.
1763 *
1764 * @returns Flags specified. (0 is possible!)
1765 * @param psz Start of Suffix. (Either dot or equal sign.)
1766 */
1767static unsigned rtlogGroupFlags(const char *psz)
1768{
1769 unsigned fFlags = 0;
1770
1771 /*
1772 * Literal flags.
1773 */
1774 while (*psz == '.')
1775 {
1776 static struct
1777 {
1778 const char *pszFlag; /* lowercase!! */
1779 unsigned fFlag;
1780 } aFlags[] =
1781 {
1782 { "eo", RTLOGGRPFLAGS_ENABLED },
1783 { "enabledonly",RTLOGGRPFLAGS_ENABLED },
1784 { "e", RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 | RTLOGGRPFLAGS_WARN },
1785 { "enabled", RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 | RTLOGGRPFLAGS_WARN },
1786 { "l1", RTLOGGRPFLAGS_LEVEL_1 },
1787 { "level1", RTLOGGRPFLAGS_LEVEL_1 },
1788 { "l", RTLOGGRPFLAGS_LEVEL_2 },
1789 { "l2", RTLOGGRPFLAGS_LEVEL_2 },
1790 { "level2", RTLOGGRPFLAGS_LEVEL_2 },
1791 { "l3", RTLOGGRPFLAGS_LEVEL_3 },
1792 { "level3", RTLOGGRPFLAGS_LEVEL_3 },
1793 { "l4", RTLOGGRPFLAGS_LEVEL_4 },
1794 { "level4", RTLOGGRPFLAGS_LEVEL_4 },
1795 { "l5", RTLOGGRPFLAGS_LEVEL_5 },
1796 { "level5", RTLOGGRPFLAGS_LEVEL_5 },
1797 { "l6", RTLOGGRPFLAGS_LEVEL_6 },
1798 { "level6", RTLOGGRPFLAGS_LEVEL_6 },
1799 { "l7", RTLOGGRPFLAGS_LEVEL_7 },
1800 { "level7", RTLOGGRPFLAGS_LEVEL_7 },
1801 { "l8", RTLOGGRPFLAGS_LEVEL_8 },
1802 { "level8", RTLOGGRPFLAGS_LEVEL_8 },
1803 { "l9", RTLOGGRPFLAGS_LEVEL_9 },
1804 { "level9", RTLOGGRPFLAGS_LEVEL_9 },
1805 { "l10", RTLOGGRPFLAGS_LEVEL_10 },
1806 { "level10", RTLOGGRPFLAGS_LEVEL_10 },
1807 { "l11", RTLOGGRPFLAGS_LEVEL_11 },
1808 { "level11", RTLOGGRPFLAGS_LEVEL_11 },
1809 { "l12", RTLOGGRPFLAGS_LEVEL_12 },
1810 { "level12", RTLOGGRPFLAGS_LEVEL_12 },
1811 { "f", RTLOGGRPFLAGS_FLOW },
1812 { "flow", RTLOGGRPFLAGS_FLOW },
1813 { "w", RTLOGGRPFLAGS_WARN },
1814 { "warn", RTLOGGRPFLAGS_WARN },
1815 { "warning", RTLOGGRPFLAGS_WARN },
1816 { "restrict", RTLOGGRPFLAGS_RESTRICT },
1817
1818 };
1819 unsigned i;
1820 bool fFound = false;
1821 psz++;
1822 for (i = 0; i < RT_ELEMENTS(aFlags) && !fFound; i++)
1823 {
1824 const char *psz1 = aFlags[i].pszFlag;
1825 const char *psz2 = psz;
1826 while (*psz1 == RT_C_TO_LOWER(*psz2))
1827 {
1828 psz1++;
1829 psz2++;
1830 if (!*psz1)
1831 {
1832 if ( (*psz2 >= 'a' && *psz2 <= 'z')
1833 || (*psz2 >= 'A' && *psz2 <= 'Z')
1834 || (*psz2 >= '0' && *psz2 <= '9') )
1835 break;
1836 fFlags |= aFlags[i].fFlag;
1837 fFound = true;
1838 psz = psz2;
1839 break;
1840 }
1841 } /* strincmp */
1842 } /* for each flags */
1843 AssertMsg(fFound, ("%.15s...", psz));
1844 }
1845
1846 /*
1847 * Flag value.
1848 */
1849 if (*psz == '=')
1850 {
1851 psz++;
1852 if (*psz == '~')
1853 fFlags = ~RTStrToInt32(psz + 1);
1854 else
1855 fFlags = RTStrToInt32(psz);
1856 }
1857
1858 return fFlags;
1859}
1860
1861/**
1862 * Helper for RTLogGetGroupSettings.
1863 */
1864static int rtLogGetGroupSettingsAddOne(const char *pszName, uint32_t fGroup, char **ppszBuf, size_t *pcchBuf, bool *pfNotFirst)
1865{
1866# define APPEND_PSZ(psz,cch) do { memcpy(*ppszBuf, (psz), (cch)); *ppszBuf += (cch); *pcchBuf -= (cch); } while (0)
1867# define APPEND_SZ(sz) APPEND_PSZ(sz, sizeof(sz) - 1)
1868# define APPEND_CH(ch) do { **ppszBuf = (ch); *ppszBuf += 1; *pcchBuf -= 1; } while (0)
1869
1870 /*
1871 * Add the name.
1872 */
1873 size_t cchName = strlen(pszName);
1874 if (cchName + 1 + *pfNotFirst > *pcchBuf)
1875 return VERR_BUFFER_OVERFLOW;
1876 if (*pfNotFirst)
1877 APPEND_CH(' ');
1878 else
1879 *pfNotFirst = true;
1880 APPEND_PSZ(pszName, cchName);
1881
1882 /*
1883 * Only generate mnemonics for the simple+common bits.
1884 */
1885 if (fGroup == (RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1))
1886 /* nothing */;
1887 else if ( fGroup == (RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 | RTLOGGRPFLAGS_LEVEL_2 | RTLOGGRPFLAGS_FLOW)
1888 && *pcchBuf >= sizeof(".e.l.f"))
1889 APPEND_SZ(".e.l.f");
1890 else if ( fGroup == (RTLOGGRPFLAGS_ENABLED | RTLOGGRPFLAGS_LEVEL_1 | RTLOGGRPFLAGS_FLOW)
1891 && *pcchBuf >= sizeof(".e.f"))
1892 APPEND_SZ(".e.f");
1893 else if (*pcchBuf >= 1 + 10 + 1)
1894 {
1895 size_t cch;
1896 APPEND_CH('=');
1897 cch = RTStrFormatNumber(*ppszBuf, fGroup, 16, 0, 0, RTSTR_F_SPECIAL | RTSTR_F_32BIT);
1898 *ppszBuf += cch;
1899 *pcchBuf -= cch;
1900 }
1901 else
1902 return VERR_BUFFER_OVERFLOW;
1903
1904# undef APPEND_PSZ
1905# undef APPEND_SZ
1906# undef APPEND_CH
1907 return VINF_SUCCESS;
1908}
1909
1910
1911/**
1912 * Get the current log group settings as a string.
1913 *
1914 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
1915 * @param pLogger Logger instance (NULL for default logger).
1916 * @param pszBuf The output buffer.
1917 * @param cchBuf The size of the output buffer. Must be greater
1918 * than zero.
1919 */
1920RTDECL(int) RTLogGetGroupSettings(PRTLOGGER pLogger, char *pszBuf, size_t cchBuf)
1921{
1922 bool fNotFirst = false;
1923 int rc = VINF_SUCCESS;
1924 uint32_t cGroups;
1925 uint32_t fGroup;
1926 uint32_t i;
1927
1928 Assert(cchBuf);
1929
1930 /*
1931 * Resolve defaults.
1932 */
1933 if (!pLogger)
1934 {
1935 pLogger = RTLogDefaultInstance();
1936 if (!pLogger)
1937 {
1938 *pszBuf = '\0';
1939 return VINF_SUCCESS;
1940 }
1941 }
1942
1943 cGroups = pLogger->cGroups;
1944
1945 /*
1946 * Check if all are the same.
1947 */
1948 fGroup = pLogger->afGroups[0];
1949 for (i = 1; i < cGroups; i++)
1950 if (pLogger->afGroups[i] != fGroup)
1951 break;
1952 if (i >= cGroups)
1953 rc = rtLogGetGroupSettingsAddOne("all", fGroup, &pszBuf, &cchBuf, &fNotFirst);
1954 else
1955 {
1956
1957 /*
1958 * Iterate all the groups and print all that are enabled.
1959 */
1960 for (i = 0; i < cGroups; i++)
1961 {
1962 fGroup = pLogger->afGroups[i];
1963 if (fGroup)
1964 {
1965 const char *pszName = pLogger->pInt->papszGroups[i];
1966 if (pszName)
1967 {
1968 rc = rtLogGetGroupSettingsAddOne(pszName, fGroup, &pszBuf, &cchBuf, &fNotFirst);
1969 if (rc)
1970 break;
1971 }
1972 }
1973 }
1974 }
1975
1976 *pszBuf = '\0';
1977 return rc;
1978}
1979RT_EXPORT_SYMBOL(RTLogGetGroupSettings);
1980
1981#endif /* !IN_RC */
1982
1983/**
1984 * Updates the flags for the logger instance using the specified
1985 * specification string.
1986 *
1987 * @returns iprt status code.
1988 * Failures can safely be ignored.
1989 * @param pLogger Logger instance (NULL for default logger).
1990 * @param pszValue Value to parse.
1991 */
1992RTDECL(int) RTLogFlags(PRTLOGGER pLogger, const char *pszValue)
1993{
1994 int rc = VINF_SUCCESS;
1995
1996 /*
1997 * Resolve defaults.
1998 */
1999 if (!pLogger)
2000 {
2001 pLogger = RTLogDefaultInstance();
2002 if (!pLogger)
2003 return VINF_SUCCESS;
2004 }
2005
2006 /*
2007 * Iterate the string.
2008 */
2009 while (*pszValue)
2010 {
2011 /* check no prefix. */
2012 bool fNo = false;
2013 char ch;
2014 unsigned i;
2015
2016 /* skip blanks. */
2017 while (RT_C_IS_SPACE(*pszValue))
2018 pszValue++;
2019 if (!*pszValue)
2020 return rc;
2021
2022 while ((ch = *pszValue) != '\0')
2023 {
2024 if (ch == 'n' && pszValue[1] == 'o')
2025 {
2026 pszValue += 2;
2027 fNo = !fNo;
2028 }
2029 else if (ch == '+')
2030 {
2031 pszValue++;
2032 fNo = true;
2033 }
2034 else if (ch == '-' || ch == '!' || ch == '~')
2035 {
2036 pszValue++;
2037 fNo = !fNo;
2038 }
2039 else
2040 break;
2041 }
2042
2043 /* instruction. */
2044 for (i = 0; i < RT_ELEMENTS(g_aLogFlags); i++)
2045 {
2046 if (!strncmp(pszValue, g_aLogFlags[i].pszInstr, g_aLogFlags[i].cchInstr))
2047 {
2048 if (!(g_aLogFlags[i].fFixedDest & pLogger->fDestFlags))
2049 {
2050 if (fNo == g_aLogFlags[i].fInverted)
2051 pLogger->fFlags |= g_aLogFlags[i].fFlag;
2052 else
2053 pLogger->fFlags &= ~g_aLogFlags[i].fFlag;
2054 }
2055 pszValue += g_aLogFlags[i].cchInstr;
2056 break;
2057 }
2058 }
2059
2060 /* unknown instruction? */
2061 if (i >= RT_ELEMENTS(g_aLogFlags))
2062 {
2063 AssertMsgFailed(("Invalid flags! unknown instruction %.20s\n", pszValue));
2064 pszValue++;
2065 }
2066
2067 /* skip blanks and delimiters. */
2068 while (RT_C_IS_SPACE(*pszValue) || *pszValue == ';')
2069 pszValue++;
2070 } /* while more environment variable value left */
2071
2072 return rc;
2073}
2074RT_EXPORT_SYMBOL(RTLogFlags);
2075
2076
2077/**
2078 * Changes the buffering setting of the specified logger.
2079 *
2080 * This can be used for optimizing longish logging sequences.
2081 *
2082 * @returns The old state.
2083 * @param pLogger The logger instance (NULL is an alias for the
2084 * default logger).
2085 * @param fBuffered The new state.
2086 */
2087RTDECL(bool) RTLogSetBuffering(PRTLOGGER pLogger, bool fBuffered)
2088{
2089 bool fOld;
2090
2091 /*
2092 * Resolve the logger instance.
2093 */
2094 if (!pLogger)
2095 {
2096 pLogger = RTLogDefaultInstance();
2097 if (!pLogger)
2098 return false;
2099 }
2100
2101 rtlogLock(pLogger);
2102 fOld = !!(pLogger->fFlags & RTLOGFLAGS_BUFFERED);
2103 if (fBuffered)
2104 pLogger->fFlags |= RTLOGFLAGS_BUFFERED;
2105 else
2106 pLogger->fFlags &= ~RTLOGFLAGS_BUFFERED;
2107 rtlogUnlock(pLogger);
2108
2109 return fOld;
2110}
2111RT_EXPORT_SYMBOL(RTLogSetBuffering);
2112
2113
2114#ifdef IN_RING3
2115RTDECL(uint32_t) RTLogSetGroupLimit(PRTLOGGER pLogger, uint32_t cMaxEntriesPerGroup)
2116{
2117 /*
2118 * Resolve the logger instance.
2119 */
2120 if (!pLogger)
2121 {
2122 pLogger = RTLogDefaultInstance();
2123 if (!pLogger)
2124 return UINT32_MAX;
2125 }
2126
2127 rtlogLock(pLogger);
2128 uint32_t cOld = pLogger->pInt->cMaxEntriesPerGroup;
2129 pLogger->pInt->cMaxEntriesPerGroup = cMaxEntriesPerGroup;
2130 rtlogUnlock(pLogger);
2131
2132 return cOld;
2133}
2134#endif
2135
2136#ifndef IN_RC
2137
2138/**
2139 * Get the current log flags as a string.
2140 *
2141 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
2142 * @param pLogger Logger instance (NULL for default logger).
2143 * @param pszBuf The output buffer.
2144 * @param cchBuf The size of the output buffer. Must be greater
2145 * than zero.
2146 */
2147RTDECL(int) RTLogGetFlags(PRTLOGGER pLogger, char *pszBuf, size_t cchBuf)
2148{
2149 bool fNotFirst = false;
2150 int rc = VINF_SUCCESS;
2151 uint32_t fFlags;
2152 unsigned i;
2153
2154 Assert(cchBuf);
2155
2156 /*
2157 * Resolve defaults.
2158 */
2159 if (!pLogger)
2160 {
2161 pLogger = RTLogDefaultInstance();
2162 if (!pLogger)
2163 {
2164 *pszBuf = '\0';
2165 return VINF_SUCCESS;
2166 }
2167 }
2168
2169 /*
2170 * Add the flags in the list.
2171 */
2172 fFlags = pLogger->fFlags;
2173 for (i = 0; i < RT_ELEMENTS(g_aLogFlags); i++)
2174 if ( !g_aLogFlags[i].fInverted
2175 ? (g_aLogFlags[i].fFlag & fFlags)
2176 : !(g_aLogFlags[i].fFlag & fFlags))
2177 {
2178 size_t cchInstr = g_aLogFlags[i].cchInstr;
2179 if (cchInstr + fNotFirst + 1 > cchBuf)
2180 {
2181 rc = VERR_BUFFER_OVERFLOW;
2182 break;
2183 }
2184 if (fNotFirst)
2185 {
2186 *pszBuf++ = ' ';
2187 cchBuf--;
2188 }
2189 memcpy(pszBuf, g_aLogFlags[i].pszInstr, cchInstr);
2190 pszBuf += cchInstr;
2191 cchBuf -= cchInstr;
2192 fNotFirst = true;
2193 }
2194 *pszBuf = '\0';
2195 return rc;
2196}
2197RT_EXPORT_SYMBOL(RTLogGetFlags);
2198
2199
2200/**
2201 * Finds the end of a destination value.
2202 *
2203 * The value ends when we counter a ';' or a free standing word (space on both
2204 * from the g_aLogDst table. (If this is problematic for someone, we could
2205 * always do quoting and escaping.)
2206 *
2207 * @returns Value length in chars.
2208 * @param pszValue The first char after '=' or ':'.
2209 */
2210static size_t rtLogDestFindValueLength(const char *pszValue)
2211{
2212 size_t off = 0;
2213 char ch;
2214 while ((ch = pszValue[off]) != '\0' && ch != ';')
2215 {
2216 if (!RT_C_IS_SPACE(ch))
2217 off++;
2218 else
2219 {
2220 unsigned i;
2221 size_t cchThusFar = off;
2222 do
2223 off++;
2224 while ((ch = pszValue[off]) != '\0' && RT_C_IS_SPACE(ch));
2225 if (ch == ';')
2226 return cchThusFar;
2227
2228 if (ch == 'n' && pszValue[off + 1] == 'o')
2229 off += 2;
2230 for (i = 0; i < RT_ELEMENTS(g_aLogDst); i++)
2231 if (!strncmp(&pszValue[off], g_aLogDst[i].pszInstr, g_aLogDst[i].cchInstr))
2232 {
2233 ch = pszValue[off + g_aLogDst[i].cchInstr];
2234 if (ch == '\0' || RT_C_IS_SPACE(ch) || ch == '=' || ch == ':' || ch == ';')
2235 return cchThusFar;
2236 }
2237 }
2238 }
2239 return off;
2240}
2241
2242
2243/**
2244 * Updates the logger destination using the specified string.
2245 *
2246 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
2247 * @param pLogger Logger instance (NULL for default logger).
2248 * @param pszValue The value to parse.
2249 */
2250RTDECL(int) RTLogDestinations(PRTLOGGER pLogger, char const *pszValue)
2251{
2252 /*
2253 * Resolve defaults.
2254 */
2255 if (!pLogger)
2256 {
2257 pLogger = RTLogDefaultInstance();
2258 if (!pLogger)
2259 return VINF_SUCCESS;
2260 }
2261
2262 /*
2263 * Do the parsing.
2264 */
2265 while (*pszValue)
2266 {
2267 bool fNo;
2268 unsigned i;
2269
2270 /* skip blanks. */
2271 while (RT_C_IS_SPACE(*pszValue))
2272 pszValue++;
2273 if (!*pszValue)
2274 break;
2275
2276 /* check no prefix. */
2277 fNo = false;
2278 if ( pszValue[0] == 'n'
2279 && pszValue[1] == 'o'
2280 && ( pszValue[2] != 'd'
2281 || pszValue[3] != 'e'
2282 || pszValue[4] != 'n'
2283 || pszValue[5] != 'y'))
2284 {
2285 fNo = true;
2286 pszValue += 2;
2287 }
2288
2289 /* instruction. */
2290 for (i = 0; i < RT_ELEMENTS(g_aLogDst); i++)
2291 {
2292 size_t cchInstr = strlen(g_aLogDst[i].pszInstr);
2293 if (!strncmp(pszValue, g_aLogDst[i].pszInstr, cchInstr))
2294 {
2295 if (!fNo)
2296 pLogger->fDestFlags |= g_aLogDst[i].fFlag;
2297 else
2298 pLogger->fDestFlags &= ~g_aLogDst[i].fFlag;
2299 pszValue += cchInstr;
2300
2301 /* check for value. */
2302 while (RT_C_IS_SPACE(*pszValue))
2303 pszValue++;
2304 if (*pszValue == '=' || *pszValue == ':')
2305 {
2306 pszValue++;
2307 size_t cch = rtLogDestFindValueLength(pszValue);
2308 const char *pszEnd = pszValue + cch;
2309
2310# ifdef IN_RING3
2311 char szTmp[sizeof(pLogger->pInt->szFilename)];
2312# else
2313 char szTmp[32];
2314# endif
2315 if (0)
2316 { /* nothing */ }
2317#ifdef IN_RING3
2318
2319 /* log file name */
2320 else if (i == 0 /* file */ && !fNo)
2321 {
2322 if (!(pLogger->fDestFlags & RTLOGDEST_FIXED_FILE))
2323 {
2324 AssertReturn(cch < sizeof(pLogger->pInt->szFilename), VERR_OUT_OF_RANGE);
2325 memcpy(pLogger->pInt->szFilename, pszValue, cch);
2326 pLogger->pInt->szFilename[cch] = '\0';
2327 /** @todo reopen log file if pLogger->pInt->fCreated is true ... */
2328 }
2329 }
2330 /* log directory */
2331 else if (i == 1 /* dir */ && !fNo)
2332 {
2333 if (!(pLogger->fDestFlags & RTLOGDEST_FIXED_DIR))
2334 {
2335 const char *pszFile = RTPathFilename(pLogger->pInt->szFilename);
2336 size_t cchFile = pszFile ? strlen(pszFile) : 0;
2337 AssertReturn(cchFile + cch + 1 < sizeof(pLogger->pInt->szFilename), VERR_OUT_OF_RANGE);
2338 memcpy(szTmp, cchFile ? pszFile : "", cchFile + 1);
2339
2340 memcpy(pLogger->pInt->szFilename, pszValue, cch);
2341 pLogger->pInt->szFilename[cch] = '\0';
2342 RTPathStripTrailingSlash(pLogger->pInt->szFilename);
2343
2344 cch = strlen(pLogger->pInt->szFilename);
2345 pLogger->pInt->szFilename[cch++] = '/';
2346 memcpy(&pLogger->pInt->szFilename[cch], szTmp, cchFile);
2347 pLogger->pInt->szFilename[cch + cchFile] = '\0';
2348 /** @todo reopen log file if pLogger->pInt->fCreated is true ... */
2349 }
2350 }
2351 else if (i == 2 /* history */)
2352 {
2353 if (!fNo)
2354 {
2355 uint32_t cHistory = 0;
2356 int rc = RTStrCopyEx(szTmp, sizeof(szTmp), pszValue, cch);
2357 if (RT_SUCCESS(rc))
2358 rc = RTStrToUInt32Full(szTmp, 0, &cHistory);
2359 AssertMsgReturn(RT_SUCCESS(rc) && cHistory < _1M, ("Invalid history value %s (%Rrc)!\n", szTmp, rc), rc);
2360 pLogger->pInt->cHistory = cHistory;
2361 }
2362 else
2363 pLogger->pInt->cHistory = 0;
2364 }
2365 else if (i == 3 /* histsize */)
2366 {
2367 if (!fNo)
2368 {
2369 int rc = RTStrCopyEx(szTmp, sizeof(szTmp), pszValue, cch);
2370 if (RT_SUCCESS(rc))
2371 rc = RTStrToUInt64Full(szTmp, 0, &pLogger->pInt->cbHistoryFileMax);
2372 AssertMsgRCReturn(rc, ("Invalid history file size value %s (%Rrc)!\n", szTmp, rc), rc);
2373 if (pLogger->pInt->cbHistoryFileMax == 0)
2374 pLogger->pInt->cbHistoryFileMax = UINT64_MAX;
2375 }
2376 else
2377 pLogger->pInt->cbHistoryFileMax = UINT64_MAX;
2378 }
2379 else if (i == 4 /* histtime */)
2380 {
2381 if (!fNo)
2382 {
2383 int rc = RTStrCopyEx(szTmp, sizeof(szTmp), pszValue, cch);
2384 if (RT_SUCCESS(rc))
2385 rc = RTStrToUInt32Full(szTmp, 0, &pLogger->pInt->cSecsHistoryTimeSlot);
2386 AssertMsgRCReturn(rc, ("Invalid history time slot value %s (%Rrc)!\n", szTmp, rc), rc);
2387 if (pLogger->pInt->cSecsHistoryTimeSlot == 0)
2388 pLogger->pInt->cSecsHistoryTimeSlot = UINT32_MAX;
2389 }
2390 else
2391 pLogger->pInt->cSecsHistoryTimeSlot = UINT32_MAX;
2392 }
2393# endif /* IN_RING3 */
2394 else if (i == 5 /* ringbuf */ && !fNo)
2395 {
2396 int rc = RTStrCopyEx(szTmp, sizeof(szTmp), pszValue, cch);
2397 uint32_t cbRingBuf = 0;
2398 if (RT_SUCCESS(rc))
2399 rc = RTStrToUInt32Full(szTmp, 0, &cbRingBuf);
2400 AssertMsgRCReturn(rc, ("Invalid ring buffer size value '%s' (%Rrc)!\n", szTmp, rc), rc);
2401
2402 if (cbRingBuf == 0)
2403 cbRingBuf = RTLOG_RINGBUF_DEFAULT_SIZE;
2404 else if (cbRingBuf < RTLOG_RINGBUF_MIN_SIZE)
2405 cbRingBuf = RTLOG_RINGBUF_MIN_SIZE;
2406 else if (cbRingBuf > RTLOG_RINGBUF_MAX_SIZE)
2407 cbRingBuf = RTLOG_RINGBUF_MAX_SIZE;
2408 else
2409 cbRingBuf = RT_ALIGN_32(cbRingBuf, 64);
2410 rc = rtLogRingBufAdjust(pLogger, cbRingBuf, false /*fForce*/);
2411 if (RT_FAILURE(rc))
2412 return rc;
2413 }
2414 else
2415 AssertMsgFailedReturn(("Invalid destination value! %s%s doesn't take a value!\n",
2416 fNo ? "no" : "", g_aLogDst[i].pszInstr),
2417 VERR_INVALID_PARAMETER);
2418
2419 pszValue = pszEnd + (*pszEnd != '\0');
2420 }
2421 else if (i == 5 /* ringbuf */ && !fNo && !pLogger->pInt->pszRingBuf)
2422 {
2423 int rc = rtLogRingBufAdjust(pLogger, pLogger->pInt->cbRingBuf, false /*fForce*/);
2424 if (RT_FAILURE(rc))
2425 return rc;
2426 }
2427 break;
2428 }
2429 }
2430
2431 /* assert known instruction */
2432 AssertMsgReturn(i < RT_ELEMENTS(g_aLogDst),
2433 ("Invalid destination value! unknown instruction %.20s\n", pszValue),
2434 VERR_INVALID_PARAMETER);
2435
2436 /* skip blanks and delimiters. */
2437 while (RT_C_IS_SPACE(*pszValue) || *pszValue == ';')
2438 pszValue++;
2439 } /* while more environment variable value left */
2440
2441 return VINF_SUCCESS;
2442}
2443RT_EXPORT_SYMBOL(RTLogDestinations);
2444
2445
2446/**
2447 * Clear the file delay flag if set, opening the destination and flushing.
2448 *
2449 * @returns IPRT status code.
2450 * @param pLogger Logger instance (NULL for default logger).
2451 * @param pszValue The value to parse.
2452 * @param pErrInfo Where to return extended error info. Optional.
2453 */
2454RTDECL(int) RTLogClearFileDelayFlag(PRTLOGGER pLogger, PRTERRINFO pErrInfo)
2455{
2456 /*
2457 * Resolve defaults.
2458 */
2459 if (!pLogger)
2460 {
2461 pLogger = RTLogDefaultInstance();
2462 if (!pLogger)
2463 return VINF_SUCCESS;
2464 }
2465
2466 /*
2467 * Do the work.
2468 */
2469 int rc = rtlogLock(pLogger);
2470 if (RT_SUCCESS(rc))
2471 {
2472 if (pLogger->fDestFlags & RTLOGDEST_F_DELAY_FILE)
2473 {
2474 pLogger->fDestFlags &= ~RTLOGDEST_F_DELAY_FILE;
2475# ifdef IN_RING3
2476 if ( pLogger->fDestFlags & RTLOGDEST_FILE
2477 && pLogger->pInt->hFile == NIL_RTFILE)
2478 {
2479 rc = rtR3LogOpenFileDestination(pLogger, pErrInfo);
2480 if (RT_SUCCESS(rc))
2481 rtlogFlush(pLogger, false /*fNeedSpace*/);
2482 }
2483# endif
2484 RT_NOREF(pErrInfo); /** @todo fix create API to use RTErrInfo */
2485 }
2486 rtlogUnlock(pLogger);
2487 }
2488 return VINF_SUCCESS;
2489}
2490RT_EXPORT_SYMBOL(RTLogClearFileDelayFlag);
2491
2492
2493/**
2494 * Get the current log destinations as a string.
2495 *
2496 * @returns VINF_SUCCESS or VERR_BUFFER_OVERFLOW.
2497 * @param pLogger Logger instance (NULL for default logger).
2498 * @param pszBuf The output buffer.
2499 * @param cchBuf The size of the output buffer. Must be greater
2500 * than 0.
2501 */
2502RTDECL(int) RTLogGetDestinations(PRTLOGGER pLogger, char *pszBuf, size_t cchBuf)
2503{
2504 bool fNotFirst = false;
2505 int rc = VINF_SUCCESS;
2506 uint32_t fDestFlags;
2507 unsigned i;
2508
2509 AssertReturn(cchBuf, VERR_INVALID_PARAMETER);
2510 *pszBuf = '\0';
2511
2512 /*
2513 * Resolve defaults.
2514 */
2515 if (!pLogger)
2516 {
2517 pLogger = RTLogDefaultInstance();
2518 if (!pLogger)
2519 return VINF_SUCCESS;
2520 }
2521
2522 /*
2523 * Add the flags in the list.
2524 */
2525 fDestFlags = pLogger->fDestFlags;
2526 for (i = 6; i < RT_ELEMENTS(g_aLogDst); i++)
2527 if (g_aLogDst[i].fFlag & fDestFlags)
2528 {
2529 if (fNotFirst)
2530 {
2531 rc = RTStrCopyP(&pszBuf, &cchBuf, " ");
2532 if (RT_FAILURE(rc))
2533 return rc;
2534 }
2535 rc = RTStrCopyP(&pszBuf, &cchBuf, g_aLogDst[i].pszInstr);
2536 if (RT_FAILURE(rc))
2537 return rc;
2538 fNotFirst = true;
2539 }
2540
2541 char szNum[32];
2542
2543# ifdef IN_RING3
2544 /*
2545 * Add the filename.
2546 */
2547 if (fDestFlags & RTLOGDEST_FILE)
2548 {
2549 rc = RTStrCopyP(&pszBuf, &cchBuf, fNotFirst ? " file=" : "file=");
2550 if (RT_FAILURE(rc))
2551 return rc;
2552 rc = RTStrCopyP(&pszBuf, &cchBuf, pLogger->pInt->szFilename);
2553 if (RT_FAILURE(rc))
2554 return rc;
2555 fNotFirst = true;
2556
2557 if (pLogger->pInt->cHistory)
2558 {
2559 RTStrPrintf(szNum, sizeof(szNum), fNotFirst ? " history=%u" : "history=%u", pLogger->pInt->cHistory);
2560 rc = RTStrCopyP(&pszBuf, &cchBuf, szNum);
2561 if (RT_FAILURE(rc))
2562 return rc;
2563 fNotFirst = true;
2564 }
2565 if (pLogger->pInt->cbHistoryFileMax != UINT64_MAX)
2566 {
2567 RTStrPrintf(szNum, sizeof(szNum), fNotFirst ? " histsize=%llu" : "histsize=%llu", pLogger->pInt->cbHistoryFileMax);
2568 rc = RTStrCopyP(&pszBuf, &cchBuf, szNum);
2569 if (RT_FAILURE(rc))
2570 return rc;
2571 fNotFirst = true;
2572 }
2573 if (pLogger->pInt->cSecsHistoryTimeSlot != UINT32_MAX)
2574 {
2575 RTStrPrintf(szNum, sizeof(szNum), fNotFirst ? " histtime=%llu" : "histtime=%llu", pLogger->pInt->cSecsHistoryTimeSlot);
2576 rc = RTStrCopyP(&pszBuf, &cchBuf, szNum);
2577 if (RT_FAILURE(rc))
2578 return rc;
2579 fNotFirst = true;
2580 }
2581 }
2582# endif /* IN_RING3 */
2583
2584 /*
2585 * Add the ring buffer.
2586 */
2587 if (fDestFlags & RTLOGDEST_RINGBUF)
2588 {
2589 if (pLogger->pInt->cbRingBuf == RTLOG_RINGBUF_DEFAULT_SIZE)
2590 rc = RTStrCopyP(&pszBuf, &cchBuf, fNotFirst ? " ringbuf" : "ringbuf");
2591 else
2592 {
2593 RTStrPrintf(szNum, sizeof(szNum), fNotFirst ? " ringbuf=%#x" : "ringbuf=%#x", pLogger->pInt->cbRingBuf);
2594 rc = RTStrCopyP(&pszBuf, &cchBuf, szNum);
2595 }
2596 if (RT_FAILURE(rc))
2597 return rc;
2598 fNotFirst = true;
2599 }
2600
2601 return VINF_SUCCESS;
2602}
2603RT_EXPORT_SYMBOL(RTLogGetDestinations);
2604
2605#endif /* !IN_RC */
2606
2607/**
2608 * Flushes the specified logger.
2609 *
2610 * @param pLogger The logger instance to flush.
2611 * If NULL the default instance is used. The default instance
2612 * will not be initialized by this call.
2613 */
2614RTDECL(void) RTLogFlush(PRTLOGGER pLogger)
2615{
2616 /*
2617 * Resolve defaults.
2618 */
2619 if (!pLogger)
2620 {
2621#ifdef IN_RC
2622 pLogger = &g_Logger;
2623#else
2624 pLogger = g_pLogger;
2625#endif
2626 if (!pLogger)
2627 return;
2628 }
2629
2630 /*
2631 * Any thing to flush?
2632 */
2633 if ( pLogger->offScratch
2634#ifndef IN_RC
2635 || (pLogger->fDestFlags & RTLOGDEST_RINGBUF)
2636#endif
2637 )
2638 {
2639#ifndef IN_RC
2640 /*
2641 * Acquire logger instance sem.
2642 */
2643 int rc = rtlogLock(pLogger);
2644 if (RT_FAILURE(rc))
2645 return;
2646#endif
2647 /*
2648 * Call worker.
2649 */
2650 rtlogFlush(pLogger, false /*fNeedSpace*/);
2651
2652#ifndef IN_RC
2653 /*
2654 * Since this is an explicit flush call, the ring buffer content should
2655 * be flushed to the other destinations if active.
2656 */
2657 if ( (pLogger->fDestFlags & RTLOGDEST_RINGBUF)
2658 && pLogger->pInt->pszRingBuf /* paranoia */)
2659 rtLogRingBufFlush(pLogger);
2660
2661 /*
2662 * Release the semaphore.
2663 */
2664 rtlogUnlock(pLogger);
2665#endif
2666 }
2667}
2668RT_EXPORT_SYMBOL(RTLogFlush);
2669
2670
2671/**
2672 * Common worker for RTLogDefaultInstance and RTLogDefaultInstanceEx.
2673 */
2674DECL_FORCE_INLINE(PRTLOGGER) rtLogDefaultInstanceCommon(void)
2675{
2676#ifdef IN_RC
2677 return &g_Logger;
2678
2679#else /* !IN_RC */
2680# ifdef IN_RING0
2681 /*
2682 * Check per thread loggers first.
2683 */
2684 if (g_cPerThreadLoggers)
2685 {
2686 const RTNATIVETHREAD Self = RTThreadNativeSelf();
2687 int32_t i = RT_ELEMENTS(g_aPerThreadLoggers);
2688 while (i-- > 0)
2689 if (g_aPerThreadLoggers[i].NativeThread == Self)
2690 return g_aPerThreadLoggers[i].pLogger;
2691 }
2692# endif /* IN_RING0 */
2693
2694 /*
2695 * If no per thread logger, use the default one.
2696 */
2697 if (!g_pLogger)
2698 g_pLogger = RTLogDefaultInit();
2699 return g_pLogger;
2700#endif /* !IN_RC */
2701}
2702
2703
2704RTDECL(PRTLOGGER) RTLogDefaultInstance(void)
2705{
2706 return rtLogDefaultInstanceCommon();
2707}
2708RT_EXPORT_SYMBOL(RTLogDefaultInstance);
2709
2710
2711RTDECL(PRTLOGGER) RTLogDefaultInstanceEx(uint32_t fFlagsAndGroup)
2712{
2713 PRTLOGGER pLogger = rtLogDefaultInstanceCommon();
2714 if (pLogger)
2715 {
2716 if (pLogger->fFlags & RTLOGFLAGS_DISABLED)
2717 pLogger = NULL;
2718 else
2719 {
2720 uint16_t const fFlags = RT_LO_U16(fFlagsAndGroup);
2721 uint16_t const iGroup = RT_HI_U16(fFlagsAndGroup);
2722 if ( iGroup != UINT16_MAX
2723 && ( (pLogger->afGroups[iGroup < pLogger->cGroups ? iGroup : 0] & (fFlags | (uint32_t)RTLOGGRPFLAGS_ENABLED))
2724 != (fFlags | (uint32_t)RTLOGGRPFLAGS_ENABLED)))
2725 pLogger = NULL;
2726 }
2727 }
2728 return pLogger;
2729}
2730RT_EXPORT_SYMBOL(RTLogDefaultInstanceEx);
2731
2732
2733/**
2734 * Common worker for RTLogGetDefaultInstance and RTLogGetDefaultInstanceEx.
2735 */
2736DECL_FORCE_INLINE(PRTLOGGER) rtLogGetDefaultInstanceCommon(void)
2737{
2738#ifdef IN_RC
2739 return &g_Logger;
2740#else
2741# ifdef IN_RING0
2742 /*
2743 * Check per thread loggers first.
2744 */
2745 if (g_cPerThreadLoggers)
2746 {
2747 const RTNATIVETHREAD Self = RTThreadNativeSelf();
2748 int32_t i = RT_ELEMENTS(g_aPerThreadLoggers);
2749 while (i-- > 0)
2750 if (g_aPerThreadLoggers[i].NativeThread == Self)
2751 return g_aPerThreadLoggers[i].pLogger;
2752 }
2753# endif /* IN_RING0 */
2754
2755 return g_pLogger;
2756#endif
2757}
2758
2759
2760RTDECL(PRTLOGGER) RTLogGetDefaultInstance(void)
2761{
2762 return rtLogGetDefaultInstanceCommon();
2763}
2764RT_EXPORT_SYMBOL(RTLogGetDefaultInstance);
2765
2766
2767RTDECL(PRTLOGGER) RTLogGetDefaultInstanceEx(uint32_t fFlagsAndGroup)
2768{
2769 PRTLOGGER pLogger = rtLogGetDefaultInstanceCommon();
2770 if (pLogger)
2771 {
2772 if (pLogger->fFlags & RTLOGFLAGS_DISABLED)
2773 pLogger = NULL;
2774 else
2775 {
2776 uint32_t const fFlags = RT_LO_U16(fFlagsAndGroup);
2777 uint16_t const iGroup = RT_HI_U16(fFlagsAndGroup);
2778 if ( iGroup != UINT16_MAX
2779 && ( (pLogger->afGroups[iGroup < pLogger->cGroups ? iGroup : 0] & (fFlags | RTLOGGRPFLAGS_ENABLED))
2780 != (fFlags | RTLOGGRPFLAGS_ENABLED)))
2781 pLogger = NULL;
2782 }
2783 }
2784 return pLogger;
2785}
2786RT_EXPORT_SYMBOL(RTLogGetDefaultInstanceEx);
2787
2788
2789#ifndef IN_RC
2790/**
2791 * Sets the default logger instance.
2792 *
2793 * @returns iprt status code.
2794 * @param pLogger The new default logger instance.
2795 */
2796RTDECL(PRTLOGGER) RTLogSetDefaultInstance(PRTLOGGER pLogger)
2797{
2798 return ASMAtomicXchgPtrT(&g_pLogger, pLogger, PRTLOGGER);
2799}
2800RT_EXPORT_SYMBOL(RTLogSetDefaultInstance);
2801#endif /* !IN_RC */
2802
2803
2804#ifdef IN_RING0
2805/**
2806 * Changes the default logger instance for the current thread.
2807 *
2808 * @returns IPRT status code.
2809 * @param pLogger The logger instance. Pass NULL for deregistration.
2810 * @param uKey Associated key for cleanup purposes. If pLogger is NULL,
2811 * all instances with this key will be deregistered. So in
2812 * order to only deregister the instance associated with the
2813 * current thread use 0.
2814 */
2815RTDECL(int) RTLogSetDefaultInstanceThread(PRTLOGGER pLogger, uintptr_t uKey)
2816{
2817 int rc;
2818 RTNATIVETHREAD Self = RTThreadNativeSelf();
2819 if (pLogger)
2820 {
2821 int32_t i;
2822 unsigned j;
2823
2824 AssertReturn(pLogger->u32Magic == RTLOGGER_MAGIC, VERR_INVALID_MAGIC);
2825
2826 /*
2827 * Iterate the table to see if there is already an entry for this thread.
2828 */
2829 i = RT_ELEMENTS(g_aPerThreadLoggers);
2830 while (i-- > 0)
2831 if (g_aPerThreadLoggers[i].NativeThread == Self)
2832 {
2833 ASMAtomicWritePtr((void * volatile *)&g_aPerThreadLoggers[i].uKey, (void *)uKey);
2834 g_aPerThreadLoggers[i].pLogger = pLogger;
2835 return VINF_SUCCESS;
2836 }
2837
2838 /*
2839 * Allocate a new table entry.
2840 */
2841 i = ASMAtomicIncS32(&g_cPerThreadLoggers);
2842 if (i > (int32_t)RT_ELEMENTS(g_aPerThreadLoggers))
2843 {
2844 ASMAtomicDecS32(&g_cPerThreadLoggers);
2845 return VERR_BUFFER_OVERFLOW; /* horrible error code! */
2846 }
2847
2848 for (j = 0; j < 10; j++)
2849 {
2850 i = RT_ELEMENTS(g_aPerThreadLoggers);
2851 while (i-- > 0)
2852 {
2853 AssertCompile(sizeof(RTNATIVETHREAD) == sizeof(void*));
2854 if ( g_aPerThreadLoggers[i].NativeThread == NIL_RTNATIVETHREAD
2855 && ASMAtomicCmpXchgPtr((void * volatile *)&g_aPerThreadLoggers[i].NativeThread, (void *)Self, (void *)NIL_RTNATIVETHREAD))
2856 {
2857 ASMAtomicWritePtr((void * volatile *)&g_aPerThreadLoggers[i].uKey, (void *)uKey);
2858 ASMAtomicWritePtr(&g_aPerThreadLoggers[i].pLogger, pLogger);
2859 return VINF_SUCCESS;
2860 }
2861 }
2862 }
2863
2864 ASMAtomicDecS32(&g_cPerThreadLoggers);
2865 rc = VERR_INTERNAL_ERROR;
2866 }
2867 else
2868 {
2869 /*
2870 * Search the array for the current thread.
2871 */
2872 int32_t i = RT_ELEMENTS(g_aPerThreadLoggers);
2873 while (i-- > 0)
2874 if ( g_aPerThreadLoggers[i].NativeThread == Self
2875 || g_aPerThreadLoggers[i].uKey == uKey)
2876 {
2877 ASMAtomicWriteNullPtr((void * volatile *)&g_aPerThreadLoggers[i].uKey);
2878 ASMAtomicWriteNullPtr(&g_aPerThreadLoggers[i].pLogger);
2879 ASMAtomicWriteHandle(&g_aPerThreadLoggers[i].NativeThread, NIL_RTNATIVETHREAD);
2880 ASMAtomicDecS32(&g_cPerThreadLoggers);
2881 }
2882
2883 rc = VINF_SUCCESS;
2884 }
2885 return rc;
2886}
2887RT_EXPORT_SYMBOL(RTLogSetDefaultInstanceThread);
2888#endif /* IN_RING0 */
2889
2890
2891/**
2892 * Write to a logger instance.
2893 *
2894 * @param pLogger Pointer to logger instance.
2895 * @param pszFormat Format string.
2896 * @param args Format arguments.
2897 */
2898RTDECL(void) RTLogLoggerV(PRTLOGGER pLogger, const char *pszFormat, va_list args)
2899{
2900 RTLogLoggerExV(pLogger, 0, ~0U, pszFormat, args);
2901}
2902RT_EXPORT_SYMBOL(RTLogLoggerV);
2903
2904
2905/**
2906 * Write to a logger instance.
2907 *
2908 * This function will check whether the instance, group and flags makes up a
2909 * logging kind which is currently enabled before writing anything to the log.
2910 *
2911 * @param pLogger Pointer to logger instance. If NULL the default logger instance will be attempted.
2912 * @param fFlags The logging flags.
2913 * @param iGroup The group.
2914 * The value ~0U is reserved for compatibility with RTLogLogger[V] and is
2915 * only for internal usage!
2916 * @param pszFormat Format string.
2917 * @param args Format arguments.
2918 */
2919RTDECL(void) RTLogLoggerExV(PRTLOGGER pLogger, unsigned fFlags, unsigned iGroup, const char *pszFormat, va_list args)
2920{
2921 int rc;
2922
2923 /*
2924 * A NULL logger means default instance.
2925 */
2926 if (!pLogger)
2927 {
2928 pLogger = RTLogDefaultInstance();
2929 if (!pLogger)
2930 return;
2931 }
2932
2933 /*
2934 * Validate and correct iGroup.
2935 */
2936 if (iGroup != ~0U && iGroup >= pLogger->cGroups)
2937 iGroup = 0;
2938
2939 /*
2940 * If no output, then just skip it.
2941 */
2942 if ( (pLogger->fFlags & RTLOGFLAGS_DISABLED)
2943#ifndef IN_RC
2944 || !pLogger->fDestFlags
2945#endif
2946 || !pszFormat || !*pszFormat)
2947 return;
2948 if ( iGroup != ~0U
2949 && (pLogger->afGroups[iGroup] & (fFlags | RTLOGGRPFLAGS_ENABLED)) != (fFlags | RTLOGGRPFLAGS_ENABLED))
2950 return;
2951
2952 /*
2953 * Acquire logger instance sem.
2954 */
2955 rc = rtlogLock(pLogger);
2956 if (RT_FAILURE(rc))
2957 {
2958#ifdef IN_RING0
2959 if (pLogger->fDestFlags & ~RTLOGDEST_FILE)
2960 rtR0LogLoggerExFallback(pLogger->fDestFlags, pLogger->fFlags, pLogger->pInt, pszFormat, args);
2961#endif
2962 return;
2963 }
2964
2965 /*
2966 * Check restrictions and call worker.
2967 */
2968#ifndef IN_RC
2969 if (RT_UNLIKELY( (pLogger->fFlags & RTLOGFLAGS_RESTRICT_GROUPS)
2970 && iGroup < pLogger->cGroups
2971 && (pLogger->afGroups[iGroup] & RTLOGGRPFLAGS_RESTRICT)
2972 && ++pLogger->pInt->pacEntriesPerGroup[iGroup] >= pLogger->pInt->cMaxEntriesPerGroup ))
2973 {
2974 uint32_t cEntries = pLogger->pInt->pacEntriesPerGroup[iGroup];
2975 if (cEntries > pLogger->pInt->cMaxEntriesPerGroup)
2976 pLogger->pInt->pacEntriesPerGroup[iGroup] = cEntries - 1;
2977 else
2978 {
2979 rtlogLoggerExVLocked(pLogger, fFlags, iGroup, pszFormat, args);
2980 if ( pLogger->pInt->papszGroups
2981 && pLogger->pInt->papszGroups[iGroup])
2982 rtlogLoggerExFLocked(pLogger, fFlags, iGroup, "%u messages from group %s (#%u), muting it.\n",
2983 cEntries, pLogger->pInt->papszGroups[iGroup], iGroup);
2984 else
2985 rtlogLoggerExFLocked(pLogger, fFlags, iGroup, "%u messages from group #%u, muting it.\n",
2986 cEntries, iGroup);
2987 }
2988 }
2989 else
2990#endif
2991 rtlogLoggerExVLocked(pLogger, fFlags, iGroup, pszFormat, args);
2992
2993 /*
2994 * Release the semaphore.
2995 */
2996 rtlogUnlock(pLogger);
2997}
2998RT_EXPORT_SYMBOL(RTLogLoggerExV);
2999
3000
3001#ifdef IN_RING0
3002/**
3003 * For rtR0LogLoggerExFallbackOutput and rtR0LogLoggerExFallbackFlush.
3004 */
3005typedef struct RTR0LOGLOGGERFALLBACK
3006{
3007 /** The current scratch buffer offset. */
3008 uint32_t offScratch;
3009 /** The destination flags. */
3010 uint32_t fDestFlags;
3011 /** For ring buffer output. */
3012 PRTLOGGERINTERNAL pInt;
3013 /** The scratch buffer. */
3014 char achScratch[80];
3015} RTR0LOGLOGGERFALLBACK;
3016/** Pointer to RTR0LOGLOGGERFALLBACK which is used by
3017 * rtR0LogLoggerExFallbackOutput. */
3018typedef RTR0LOGLOGGERFALLBACK *PRTR0LOGLOGGERFALLBACK;
3019
3020
3021/**
3022 * Flushes the fallback buffer.
3023 *
3024 * @param pThis The scratch buffer.
3025 */
3026static void rtR0LogLoggerExFallbackFlush(PRTR0LOGLOGGERFALLBACK pThis)
3027{
3028 if (!pThis->offScratch)
3029 return;
3030
3031 if ( (pThis->fDestFlags & RTLOGDEST_RINGBUF)
3032 && pThis->pInt
3033 && pThis->pInt->pszRingBuf /* paranoia */)
3034 rtLogRingBufWrite(pThis->pInt, pThis->achScratch, pThis->offScratch);
3035 else
3036 {
3037 if (pThis->fDestFlags & RTLOGDEST_USER)
3038 RTLogWriteUser(pThis->achScratch, pThis->offScratch);
3039
3040 if (pThis->fDestFlags & RTLOGDEST_DEBUGGER)
3041 RTLogWriteDebugger(pThis->achScratch, pThis->offScratch);
3042
3043 if (pThis->fDestFlags & RTLOGDEST_STDOUT)
3044 RTLogWriteStdOut(pThis->achScratch, pThis->offScratch);
3045
3046 if (pThis->fDestFlags & RTLOGDEST_STDERR)
3047 RTLogWriteStdErr(pThis->achScratch, pThis->offScratch);
3048
3049# ifndef LOG_NO_COM
3050 if (pThis->fDestFlags & RTLOGDEST_COM)
3051 RTLogWriteCom(pThis->achScratch, pThis->offScratch);
3052# endif
3053 }
3054
3055 /* empty the buffer. */
3056 pThis->offScratch = 0;
3057}
3058
3059
3060/**
3061 * Callback for RTLogFormatV used by rtR0LogLoggerExFallback.
3062 * See PFNLOGOUTPUT() for details.
3063 */
3064static DECLCALLBACK(size_t) rtR0LogLoggerExFallbackOutput(void *pv, const char *pachChars, size_t cbChars)
3065{
3066 PRTR0LOGLOGGERFALLBACK pThis = (PRTR0LOGLOGGERFALLBACK)pv;
3067 if (cbChars)
3068 {
3069 size_t cbRet = 0;
3070 for (;;)
3071 {
3072 /* how much */
3073 uint32_t cb = sizeof(pThis->achScratch) - pThis->offScratch - 1; /* minus 1 - for the string terminator. */
3074 if (cb > cbChars)
3075 cb = (uint32_t)cbChars;
3076
3077 /* copy */
3078 memcpy(&pThis->achScratch[pThis->offScratch], pachChars, cb);
3079
3080 /* advance */
3081 pThis->offScratch += cb;
3082 cbRet += cb;
3083 cbChars -= cb;
3084
3085 /* done? */
3086 if (cbChars <= 0)
3087 return cbRet;
3088
3089 pachChars += cb;
3090
3091 /* flush */
3092 pThis->achScratch[pThis->offScratch] = '\0';
3093 rtR0LogLoggerExFallbackFlush(pThis);
3094 }
3095
3096 /* won't ever get here! */
3097 }
3098 else
3099 {
3100 /*
3101 * Termination call, flush the log.
3102 */
3103 pThis->achScratch[pThis->offScratch] = '\0';
3104 rtR0LogLoggerExFallbackFlush(pThis);
3105 return 0;
3106 }
3107}
3108
3109
3110/**
3111 * Ring-0 fallback for cases where we're unable to grab the lock.
3112 *
3113 * This will happen when we're at a too high IRQL on Windows for instance and
3114 * needs to be dealt with or we'll drop a lot of log output. This fallback will
3115 * only output to some of the log destinations as a few of them may be doing
3116 * dangerous things. We won't be doing any prefixing here either, at least not
3117 * for the present, because it's too much hassle.
3118 *
3119 * @param fDestFlags The destination flags.
3120 * @param fFlags The logger flags.
3121 * @param pInt The internal logger data, for ring buffer output.
3122 * @param pszFormat The format string.
3123 * @param va The format arguments.
3124 */
3125static void rtR0LogLoggerExFallback(uint32_t fDestFlags, uint32_t fFlags, PRTLOGGERINTERNAL pInt,
3126 const char *pszFormat, va_list va)
3127{
3128 RTR0LOGLOGGERFALLBACK This;
3129 This.fDestFlags = fDestFlags;
3130 This.pInt = pInt;
3131
3132 /* fallback indicator. */
3133 This.offScratch = 2;
3134 This.achScratch[0] = '[';
3135 This.achScratch[1] = 'F';
3136
3137 /* selected prefixes */
3138 if (fFlags & RTLOGFLAGS_PREFIX_PID)
3139 {
3140 RTPROCESS Process = RTProcSelf();
3141 This.achScratch[This.offScratch++] = ' ';
3142 This.offScratch += RTStrFormatNumber(&This.achScratch[This.offScratch], Process, 16, sizeof(RTPROCESS) * 2, 0, RTSTR_F_ZEROPAD);
3143 }
3144 if (fFlags & RTLOGFLAGS_PREFIX_TID)
3145 {
3146 RTNATIVETHREAD Thread = RTThreadNativeSelf();
3147 This.achScratch[This.offScratch++] = ' ';
3148 This.offScratch += RTStrFormatNumber(&This.achScratch[This.offScratch], Thread, 16, sizeof(RTNATIVETHREAD) * 2, 0, RTSTR_F_ZEROPAD);
3149 }
3150
3151 This.achScratch[This.offScratch++] = ']';
3152 This.achScratch[This.offScratch++] = ' ';
3153
3154 RTLogFormatV(rtR0LogLoggerExFallbackOutput, &This, pszFormat, va);
3155}
3156#endif /* IN_RING0 */
3157
3158
3159/**
3160 * vprintf like function for writing to the default log.
3161 *
3162 * @param pszFormat Printf like format string.
3163 * @param va Optional arguments as specified in pszFormat.
3164 *
3165 * @remark The API doesn't support formatting of floating point numbers at the moment.
3166 */
3167RTDECL(void) RTLogPrintfV(const char *pszFormat, va_list va)
3168{
3169 RTLogLoggerV(NULL, pszFormat, va);
3170}
3171RT_EXPORT_SYMBOL(RTLogPrintfV);
3172
3173
3174/**
3175 * Dumper vprintf-like function outputting to a logger.
3176 *
3177 * @param pvUser Pointer to the logger instance to use, NULL for
3178 * default instance.
3179 * @param pszFormat Format string.
3180 * @param va Format arguments.
3181 */
3182RTDECL(void) RTLogDumpPrintfV(void *pvUser, const char *pszFormat, va_list va)
3183{
3184 RTLogLoggerV((PRTLOGGER)pvUser, pszFormat, va);
3185}
3186RT_EXPORT_SYMBOL(RTLogDumpPrintfV);
3187
3188
3189#ifdef IN_RING3
3190
3191/**
3192 * Opens/creates the log file.
3193 *
3194 * @param pLogger The logger instance to update. NULL is not allowed!
3195 * @param pErrInfo Where to return extended error information.
3196 * Optional.
3197 */
3198static int rtlogFileOpen(PRTLOGGER pLogger, PRTERRINFO pErrInfo)
3199{
3200 uint32_t fOpen = RTFILE_O_WRITE | RTFILE_O_DENY_NONE;
3201 if (pLogger->fFlags & RTLOGFLAGS_APPEND)
3202 fOpen |= RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND;
3203 else
3204 {
3205 RTFileDelete(pLogger->pInt->szFilename);
3206 fOpen |= RTFILE_O_CREATE;
3207 }
3208 if (pLogger->fFlags & RTLOGFLAGS_WRITE_THROUGH)
3209 fOpen |= RTFILE_O_WRITE_THROUGH;
3210 if (pLogger->fDestFlags & RTLOGDEST_F_NO_DENY)
3211 fOpen = (fOpen & ~RTFILE_O_DENY_NONE) | RTFILE_O_DENY_NOT_DELETE;
3212
3213 unsigned cBackoff = 0;
3214 int rc = RTFileOpen(&pLogger->pInt->hFile, pLogger->pInt->szFilename, fOpen);
3215 while ( ( rc == VERR_SHARING_VIOLATION
3216 || (rc == VERR_ALREADY_EXISTS && !(pLogger->fFlags & RTLOGFLAGS_APPEND)))
3217 && cBackoff < RT_ELEMENTS(g_acMsLogBackoff))
3218 {
3219 RTThreadSleep(g_acMsLogBackoff[cBackoff++]);
3220 if (!(pLogger->fFlags & RTLOGFLAGS_APPEND))
3221 RTFileDelete(pLogger->pInt->szFilename);
3222 rc = RTFileOpen(&pLogger->pInt->hFile, pLogger->pInt->szFilename, fOpen);
3223 }
3224 if (RT_SUCCESS(rc))
3225 {
3226 rc = RTFileQuerySize(pLogger->pInt->hFile, &pLogger->pInt->cbHistoryFileWritten);
3227 if (RT_FAILURE(rc))
3228 {
3229 /* Don't complain if this fails, assume the file is empty. */
3230 pLogger->pInt->cbHistoryFileWritten = 0;
3231 rc = VINF_SUCCESS;
3232 }
3233 }
3234 else
3235 {
3236 pLogger->pInt->hFile = NIL_RTFILE;
3237 RTErrInfoSetF(pErrInfo, rc, N_("could not open file '%s' (fOpen=%#x)"), pLogger->pInt->szFilename, fOpen);
3238 }
3239 return rc;
3240}
3241
3242
3243/**
3244 * Closes, rotates and opens the log files if necessary.
3245 *
3246 * Used by the rtlogFlush() function as well as RTLogCreateExV.
3247 *
3248 * @param pLogger The logger instance to update. NULL is not allowed!
3249 * @param uTimeSlot Current time slot (for tikme based rotation).
3250 * @param fFirst Flag whether this is the beginning of logging, i.e.
3251 * called from RTLogCreateExV. Prevents pfnPhase from
3252 * being called.
3253 * @param pErrInfo Where to return extended error information. Optional.
3254 */
3255static void rtlogRotate(PRTLOGGER pLogger, uint32_t uTimeSlot, bool fFirst, PRTERRINFO pErrInfo)
3256{
3257 /* Suppress rotating empty log files simply because the time elapsed. */
3258 if (RT_UNLIKELY(!pLogger->pInt->cbHistoryFileWritten))
3259 pLogger->pInt->uHistoryTimeSlotStart = uTimeSlot;
3260
3261 /* Check rotation condition: file still small enough and not too old? */
3262 if (RT_LIKELY( pLogger->pInt->cbHistoryFileWritten < pLogger->pInt->cbHistoryFileMax
3263 && uTimeSlot == pLogger->pInt->uHistoryTimeSlotStart))
3264 return;
3265
3266 /*
3267 * Save "disabled" log flag and make sure logging is disabled.
3268 * The logging in the functions called during log file history
3269 * rotation would cause severe trouble otherwise.
3270 */
3271 uint32_t const fSavedFlags = pLogger->fFlags;
3272 pLogger->fFlags |= RTLOGFLAGS_DISABLED;
3273
3274 /*
3275 * Disable log rotation temporarily, otherwise with extreme settings and
3276 * chatty phase logging we could run into endless rotation.
3277 */
3278 uint32_t const cSavedHistory = pLogger->pInt->cHistory;
3279 pLogger->pInt->cHistory = 0;
3280
3281 /*
3282 * Close the old log file.
3283 */
3284 if (pLogger->pInt->hFile != NIL_RTFILE)
3285 {
3286 /* Use the callback to generate some final log contents, but only if
3287 * this is a rotation with a fully set up logger. Leave the other case
3288 * to the RTLogCreateExV function. */
3289 if (pLogger->pInt->pfnPhase && !fFirst)
3290 {
3291 uint32_t fODestFlags = pLogger->fDestFlags;
3292 pLogger->fDestFlags &= RTLOGDEST_FILE;
3293 pLogger->pInt->pfnPhase(pLogger, RTLOGPHASE_PREROTATE, rtlogPhaseMsgLocked);
3294 pLogger->fDestFlags = fODestFlags;
3295 }
3296 RTFileClose(pLogger->pInt->hFile);
3297 pLogger->pInt->hFile = NIL_RTFILE;
3298 }
3299
3300 if (cSavedHistory)
3301 {
3302 /*
3303 * Rotate the log files.
3304 */
3305 for (uint32_t i = cSavedHistory - 1; i + 1 > 0; i--)
3306 {
3307 char szOldName[sizeof(pLogger->pInt->szFilename) + 32];
3308 if (i > 0)
3309 RTStrPrintf(szOldName, sizeof(szOldName), "%s.%u", pLogger->pInt->szFilename, i);
3310 else
3311 RTStrCopy(szOldName, sizeof(szOldName), pLogger->pInt->szFilename);
3312
3313 char szNewName[sizeof(pLogger->pInt->szFilename) + 32];
3314 RTStrPrintf(szNewName, sizeof(szNewName), "%s.%u", pLogger->pInt->szFilename, i + 1);
3315
3316 unsigned cBackoff = 0;
3317 int rc = RTFileRename(szOldName, szNewName, RTFILEMOVE_FLAGS_REPLACE);
3318 while ( rc == VERR_SHARING_VIOLATION
3319 && cBackoff < RT_ELEMENTS(g_acMsLogBackoff))
3320 {
3321 RTThreadSleep(g_acMsLogBackoff[cBackoff++]);
3322 rc = RTFileRename(szOldName, szNewName, RTFILEMOVE_FLAGS_REPLACE);
3323 }
3324
3325 if (rc == VERR_FILE_NOT_FOUND)
3326 RTFileDelete(szNewName);
3327 }
3328
3329 /*
3330 * Delete excess log files.
3331 */
3332 for (uint32_t i = cSavedHistory + 1; ; i++)
3333 {
3334 char szExcessName[sizeof(pLogger->pInt->szFilename) + 32];
3335 RTStrPrintf(szExcessName, sizeof(szExcessName), "%s.%u", pLogger->pInt->szFilename, i);
3336 int rc = RTFileDelete(szExcessName);
3337 if (RT_FAILURE(rc))
3338 break;
3339 }
3340 }
3341
3342 /*
3343 * Update logger state and create new log file.
3344 */
3345 pLogger->pInt->cbHistoryFileWritten = 0;
3346 pLogger->pInt->uHistoryTimeSlotStart = uTimeSlot;
3347 rtlogFileOpen(pLogger, pErrInfo);
3348
3349 /*
3350 * Use the callback to generate some initial log contents, but only if this
3351 * is a rotation with a fully set up logger. Leave the other case to the
3352 * RTLogCreateExV function.
3353 */
3354 if (pLogger->pInt->pfnPhase && !fFirst)
3355 {
3356 uint32_t const fSavedDestFlags = pLogger->fDestFlags;
3357 pLogger->fDestFlags &= RTLOGDEST_FILE;
3358 pLogger->pInt->pfnPhase(pLogger, RTLOGPHASE_POSTROTATE, rtlogPhaseMsgLocked);
3359 pLogger->fDestFlags = fSavedDestFlags;
3360 }
3361
3362 /* Restore saved values. */
3363 pLogger->pInt->cHistory = cSavedHistory;
3364 pLogger->fFlags = fSavedFlags;
3365}
3366
3367
3368/**
3369 * Worker for RTLogCreateExV and RTLogClearFileDelayFlag.
3370 *
3371 * This will later be used to reopen the file by RTLogDestinations.
3372 *
3373 * @returns IPRT status code.
3374 * @param pLogger The logger.
3375 * @param pErrInfo Where to return extended error information.
3376 * Optional.
3377 */
3378static int rtR3LogOpenFileDestination(PRTLOGGER pLogger, PRTERRINFO pErrInfo)
3379{
3380 int rc;
3381 if (pLogger->fFlags & RTLOGFLAGS_APPEND)
3382 {
3383 rc = rtlogFileOpen(pLogger, pErrInfo);
3384
3385 /* Rotate in case of appending to a too big log file,
3386 otherwise this simply doesn't do anything. */
3387 rtlogRotate(pLogger, 0, true /* fFirst */, pErrInfo);
3388 }
3389 else
3390 {
3391 /* Force rotation if it is configured. */
3392 pLogger->pInt->cbHistoryFileWritten = UINT64_MAX;
3393 rtlogRotate(pLogger, 0, true /* fFirst */, pErrInfo);
3394
3395 /* If the file is not open then rotation is not set up. */
3396 if (pLogger->pInt->hFile == NIL_RTFILE)
3397 {
3398 pLogger->pInt->cbHistoryFileWritten = 0;
3399 rc = rtlogFileOpen(pLogger, pErrInfo);
3400 }
3401 else
3402 rc = VINF_SUCCESS;
3403 }
3404 return rc;
3405}
3406
3407#endif /* IN_RING3 */
3408
3409
3410/**
3411 * Writes the buffer to the given log device without checking for buffered
3412 * data or anything.
3413 *
3414 * Used by the RTLogFlush() function.
3415 *
3416 * @param pLogger The logger instance to write to. NULL is not allowed!
3417 * @param fNeedSpace Set if the caller assumes space will be made available.
3418 */
3419static void rtlogFlush(PRTLOGGER pLogger, bool fNeedSpace)
3420{
3421 uint32_t const cchScratch = pLogger->offScratch;
3422 if (cchScratch == 0)
3423 return; /* nothing to flush. */
3424 NOREF(fNeedSpace);
3425
3426#ifndef IN_RC
3427 /*
3428 * If the ring buffer is active, the other destinations are only written
3429 * to when the ring buffer is flushed by RTLogFlush().
3430 */
3431 if ( (pLogger->fDestFlags & RTLOGDEST_RINGBUF)
3432 && pLogger->pInt
3433 && pLogger->pInt->pszRingBuf /* paraoia */)
3434 {
3435 rtLogRingBufWrite(pLogger->pInt, pLogger->achScratch, pLogger->offScratch);
3436 pLogger->offScratch = 0; /* empty the buffer. */
3437 }
3438 /*
3439 * In file delay mode, we ignore flush requests except when we're full
3440 * and the caller really needs some scratch space to get work done.
3441 */
3442 else
3443# ifdef IN_RING3
3444 if (!(pLogger->fDestFlags & RTLOGDEST_F_DELAY_FILE))
3445# endif
3446#endif
3447 {
3448 /* Make sure the string is terminated. On Windows, RTLogWriteDebugger
3449 will get upset if it isn't. */
3450 if (RT_LIKELY(cchScratch < sizeof(pLogger->achScratch)))
3451 pLogger->achScratch[cchScratch] = '\0';
3452 else
3453 AssertFailed();
3454
3455#ifndef IN_RC
3456 if (pLogger->fDestFlags & RTLOGDEST_USER)
3457 RTLogWriteUser(pLogger->achScratch, cchScratch);
3458
3459 if (pLogger->fDestFlags & RTLOGDEST_DEBUGGER)
3460 RTLogWriteDebugger(pLogger->achScratch, cchScratch);
3461
3462# ifdef IN_RING3
3463 if ((pLogger->fDestFlags & (RTLOGDEST_FILE | RTLOGDEST_RINGBUF)) == RTLOGDEST_FILE)
3464 {
3465 if (pLogger->pInt->hFile != NIL_RTFILE)
3466 {
3467 RTFileWrite(pLogger->pInt->hFile, pLogger->achScratch, cchScratch, NULL);
3468 if (pLogger->fFlags & RTLOGFLAGS_FLUSH)
3469 RTFileFlush(pLogger->pInt->hFile);
3470 }
3471 if (pLogger->pInt->cHistory)
3472 pLogger->pInt->cbHistoryFileWritten += cchScratch;
3473 }
3474# endif
3475
3476 if (pLogger->fDestFlags & RTLOGDEST_STDOUT)
3477 RTLogWriteStdOut(pLogger->achScratch, cchScratch);
3478
3479 if (pLogger->fDestFlags & RTLOGDEST_STDERR)
3480 RTLogWriteStdErr(pLogger->achScratch, cchScratch);
3481
3482# if (defined(IN_RING0) || defined(IN_RC)) && !defined(LOG_NO_COM)
3483 if (pLogger->fDestFlags & RTLOGDEST_COM)
3484 RTLogWriteCom(pLogger->achScratch, cchScratch);
3485# endif
3486#endif /* !IN_RC */
3487
3488#ifdef IN_RC
3489 if (pLogger->pfnFlush)
3490 pLogger->pfnFlush(pLogger);
3491#else
3492 if (pLogger->pInt->pfnFlush)
3493 pLogger->pInt->pfnFlush(pLogger);
3494#endif
3495
3496 /* empty the buffer. */
3497 pLogger->offScratch = 0;
3498
3499#ifdef IN_RING3
3500 /*
3501 * Rotate the log file if configured. Must be done after everything is
3502 * flushed, since this will also use logging/flushing to write the header
3503 * and footer messages.
3504 */
3505 if ( (pLogger->fDestFlags & RTLOGDEST_FILE)
3506 && pLogger->pInt->cHistory)
3507 rtlogRotate(pLogger, RTTimeProgramSecTS() / pLogger->pInt->cSecsHistoryTimeSlot, false /*fFirst*/, NULL /*pErrInfo*/);
3508#endif
3509 }
3510#ifdef IN_RING3
3511 else
3512 {
3513 /*
3514 * Delay file open but the caller really need some space. So, give him half a
3515 * buffer and insert a message indicating that we've dropped output.
3516 */
3517 uint32_t offHalf = sizeof(pLogger->achScratch) / 2;
3518 if (cchScratch > offHalf)
3519 {
3520 if (pLogger->fFlags & RTLOGFLAGS_USECRLF)
3521 pLogger->achScratch[offHalf++] = '\r';
3522 static const char s_szDropMsg[] = "\n[DROP DROP DROP]";
3523 memcpy(&pLogger->achScratch[offHalf], RT_STR_TUPLE(s_szDropMsg));
3524 offHalf += sizeof(s_szDropMsg) - 1;
3525 if (pLogger->fFlags & RTLOGFLAGS_USECRLF)
3526 pLogger->achScratch[offHalf++] = '\r';
3527 pLogger->achScratch[offHalf++] = '\n';
3528
3529 pLogger->offScratch = offHalf;
3530 }
3531 }
3532#endif
3533}
3534
3535
3536/**
3537 * Callback for RTLogFormatV which writes to the com port.
3538 * See PFNLOGOUTPUT() for details.
3539 */
3540static DECLCALLBACK(size_t) rtLogOutput(void *pv, const char *pachChars, size_t cbChars)
3541{
3542 PRTLOGGER pLogger = (PRTLOGGER)pv;
3543 if (cbChars)
3544 {
3545 size_t cbRet = 0;
3546 for (;;)
3547 {
3548#if defined(DEBUG) && defined(IN_RING3)
3549 /* sanity */
3550 if (pLogger->offScratch >= sizeof(pLogger->achScratch))
3551 {
3552 fprintf(stderr, "pLogger->offScratch >= sizeof(pLogger->achScratch) (%#x >= %#x)\n",
3553 pLogger->offScratch, (unsigned)sizeof(pLogger->achScratch));
3554 AssertBreakpoint(); AssertBreakpoint();
3555 }
3556#endif
3557
3558 /* how much */
3559 size_t cb = sizeof(pLogger->achScratch) - pLogger->offScratch - 1;
3560 if (cb > cbChars)
3561 cb = cbChars;
3562
3563 /* copy */
3564 memcpy(&pLogger->achScratch[pLogger->offScratch], pachChars, cb);
3565
3566 /* advance */
3567 pLogger->offScratch += (uint32_t)cb;
3568 cbRet += cb;
3569 cbChars -= cb;
3570
3571 /* done? */
3572 if (cbChars <= 0)
3573 return cbRet;
3574
3575 pachChars += cb;
3576
3577 /* flush */
3578 rtlogFlush(pLogger, true /*fNeedSpace*/);
3579 }
3580
3581 /* won't ever get here! */
3582 }
3583 else
3584 {
3585 /*
3586 * Termination call.
3587 * There's always space for a terminator, and it's not counted.
3588 */
3589 pLogger->achScratch[pLogger->offScratch] = '\0';
3590 return 0;
3591 }
3592}
3593
3594
3595/**
3596 * stpncpy implementation for use in rtLogOutputPrefixed w/ padding.
3597 *
3598 * @returns Pointer to the destination buffer byte following the copied string.
3599 * @param pszDst The destination buffer.
3600 * @param pszSrc The source string.
3601 * @param cchSrcMax The maximum number of characters to copy from
3602 * the string.
3603 * @param cchMinWidth The minimum field with, padd with spaces to
3604 * reach this.
3605 */
3606DECLINLINE(char *) rtLogStPNCpyPad(char *pszDst, const char *pszSrc, size_t cchSrcMax, size_t cchMinWidth)
3607{
3608 size_t cchSrc = 0;
3609 if (pszSrc)
3610 {
3611 cchSrc = strlen(pszSrc);
3612 if (cchSrc > cchSrcMax)
3613 cchSrc = cchSrcMax;
3614
3615 memcpy(pszDst, pszSrc, cchSrc);
3616 pszDst += cchSrc;
3617 }
3618 do
3619 *pszDst++ = ' ';
3620 while (cchSrc++ < cchMinWidth);
3621
3622 return pszDst;
3623}
3624
3625
3626/**
3627 * stpncpy implementation for use in rtLogOutputPrefixed w/ padding.
3628 *
3629 * @returns Pointer to the destination buffer byte following the copied string.
3630 * @param pszDst The destination buffer.
3631 * @param pszSrc The source string.
3632 * @param cchSrc The number of characters to copy from the
3633 * source. Equal or less than string length.
3634 * @param cchMinWidth The minimum field with, padd with spaces to
3635 * reach this.
3636 */
3637DECLINLINE(char *) rtLogStPNCpyPad2(char *pszDst, const char *pszSrc, size_t cchSrc, size_t cchMinWidth)
3638{
3639 Assert(pszSrc);
3640 Assert(strlen(pszSrc) >= cchSrc);
3641
3642 memcpy(pszDst, pszSrc, cchSrc);
3643 pszDst += cchSrc;
3644 do
3645 *pszDst++ = ' ';
3646 while (cchSrc++ < cchMinWidth);
3647
3648 return pszDst;
3649}
3650
3651
3652
3653/**
3654 * Callback for RTLogFormatV which writes to the logger instance.
3655 * This version supports prefixes.
3656 *
3657 * See PFNLOGOUTPUT() for details.
3658 */
3659static DECLCALLBACK(size_t) rtLogOutputPrefixed(void *pv, const char *pachChars, size_t cbChars)
3660{
3661 PRTLOGOUTPUTPREFIXEDARGS pArgs = (PRTLOGOUTPUTPREFIXEDARGS)pv;
3662 PRTLOGGER pLogger = pArgs->pLogger;
3663 if (cbChars)
3664 {
3665 size_t cbRet = 0;
3666 for (;;)
3667 {
3668 uint32_t offScratch = pLogger->offScratch;
3669 size_t cb = sizeof(pLogger->achScratch) - offScratch - 1;
3670 const char *pszNewLine;
3671 char *psz;
3672#ifdef IN_RC
3673 bool *pfPendingPrefix = &pLogger->fPendingPrefix;
3674#else
3675 bool *pfPendingPrefix = &pLogger->pInt->fPendingPrefix;
3676#endif
3677
3678 /*
3679 * Pending prefix?
3680 */
3681 if (*pfPendingPrefix)
3682 {
3683 *pfPendingPrefix = false;
3684
3685#if defined(DEBUG) && defined(IN_RING3)
3686 /* sanity */
3687 if (offScratch >= sizeof(pLogger->achScratch))
3688 {
3689 fprintf(stderr, "offScratch >= sizeof(pLogger->achScratch) (%#x >= %#x)\n",
3690 offScratch, (unsigned)sizeof(pLogger->achScratch));
3691 AssertBreakpoint(); AssertBreakpoint();
3692 }
3693#endif
3694
3695 /*
3696 * Flush the buffer if there isn't enough room for the maximum prefix config.
3697 * Max is 256, add a couple of extra bytes. See CCH_PREFIX check way below.
3698 */
3699 if (cb < 256 + 16)
3700 {
3701 rtlogFlush(pLogger, true /*fNeedSpace*/);
3702 offScratch = pLogger->offScratch;
3703 cb = sizeof(pLogger->achScratch) - offScratch - 1;
3704 }
3705
3706 /*
3707 * Write the prefixes.
3708 * psz is pointing to the current position.
3709 */
3710 psz = &pLogger->achScratch[offScratch];
3711 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TS)
3712 {
3713 uint64_t u64 = RTTimeNanoTS();
3714 int iBase = 16;
3715 unsigned int fFlags = RTSTR_F_ZEROPAD;
3716 if (pLogger->fFlags & RTLOGFLAGS_DECIMAL_TS)
3717 {
3718 iBase = 10;
3719 fFlags = 0;
3720 }
3721 if (pLogger->fFlags & RTLOGFLAGS_REL_TS)
3722 {
3723 static volatile uint64_t s_u64LastTs;
3724 uint64_t u64DiffTs = u64 - s_u64LastTs;
3725 s_u64LastTs = u64;
3726 /* We could have been preempted just before reading of s_u64LastTs by
3727 * another thread which wrote s_u64LastTs. In that case the difference
3728 * is negative which we simply ignore. */
3729 u64 = (int64_t)u64DiffTs < 0 ? 0 : u64DiffTs;
3730 }
3731 /* 1E15 nanoseconds = 11 days */
3732 psz += RTStrFormatNumber(psz, u64, iBase, 16, 0, fFlags);
3733 *psz++ = ' ';
3734 }
3735#define CCH_PREFIX_01 0 + 17
3736
3737 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TSC)
3738 {
3739#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
3740 uint64_t u64 = ASMReadTSC();
3741#else
3742 uint64_t u64 = RTTimeNanoTS();
3743#endif
3744 int iBase = 16;
3745 unsigned int fFlags = RTSTR_F_ZEROPAD;
3746 if (pLogger->fFlags & RTLOGFLAGS_DECIMAL_TS)
3747 {
3748 iBase = 10;
3749 fFlags = 0;
3750 }
3751 if (pLogger->fFlags & RTLOGFLAGS_REL_TS)
3752 {
3753 static volatile uint64_t s_u64LastTsc;
3754 int64_t i64DiffTsc = u64 - s_u64LastTsc;
3755 s_u64LastTsc = u64;
3756 /* We could have been preempted just before reading of s_u64LastTsc by
3757 * another thread which wrote s_u64LastTsc. In that case the difference
3758 * is negative which we simply ignore. */
3759 u64 = i64DiffTsc < 0 ? 0 : i64DiffTsc;
3760 }
3761 /* 1E15 ticks at 4GHz = 69 hours */
3762 psz += RTStrFormatNumber(psz, u64, iBase, 16, 0, fFlags);
3763 *psz++ = ' ';
3764 }
3765#define CCH_PREFIX_02 CCH_PREFIX_01 + 17
3766
3767 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_MS_PROG)
3768 {
3769#if defined(IN_RING3) || defined(IN_RC)
3770 uint64_t u64 = RTTimeProgramMilliTS();
3771#else
3772 uint64_t u64 = (RTTimeNanoTS() - pLogger->pInt->nsR0ProgramStart) / RT_NS_1MS;
3773#endif
3774 /* 1E8 milliseconds = 27 hours */
3775 psz += RTStrFormatNumber(psz, u64, 10, 9, 0, RTSTR_F_ZEROPAD);
3776 *psz++ = ' ';
3777 }
3778#define CCH_PREFIX_03 CCH_PREFIX_02 + 21
3779
3780 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TIME)
3781 {
3782#if defined(IN_RING3) || defined(IN_RING0)
3783 RTTIMESPEC TimeSpec;
3784 RTTIME Time;
3785 RTTimeExplode(&Time, RTTimeNow(&TimeSpec));
3786 psz += RTStrFormatNumber(psz, Time.u8Hour, 10, 2, 0, RTSTR_F_ZEROPAD);
3787 *psz++ = ':';
3788 psz += RTStrFormatNumber(psz, Time.u8Minute, 10, 2, 0, RTSTR_F_ZEROPAD);
3789 *psz++ = ':';
3790 psz += RTStrFormatNumber(psz, Time.u8Second, 10, 2, 0, RTSTR_F_ZEROPAD);
3791 *psz++ = '.';
3792 psz += RTStrFormatNumber(psz, Time.u32Nanosecond / 1000, 10, 6, 0, RTSTR_F_ZEROPAD);
3793 *psz++ = ' ';
3794#else
3795 memset(psz, ' ', 16);
3796 psz += 16;
3797#endif
3798 }
3799#define CCH_PREFIX_04 CCH_PREFIX_03 + (3+1+3+1+3+1+7+1)
3800
3801 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TIME_PROG)
3802 {
3803
3804#if defined(IN_RING3) || defined(IN_RC)
3805 uint64_t u64 = RTTimeProgramMicroTS();
3806#else
3807 uint64_t u64 = (RTTimeNanoTS() - pLogger->pInt->nsR0ProgramStart) / RT_NS_1US;
3808
3809#endif
3810 psz += RTStrFormatNumber(psz, (uint32_t)(u64 / RT_US_1HOUR), 10, 2, 0, RTSTR_F_ZEROPAD);
3811 *psz++ = ':';
3812 uint32_t u32 = (uint32_t)(u64 % RT_US_1HOUR);
3813 psz += RTStrFormatNumber(psz, u32 / RT_US_1MIN, 10, 2, 0, RTSTR_F_ZEROPAD);
3814 *psz++ = ':';
3815 u32 %= RT_US_1MIN;
3816
3817 psz += RTStrFormatNumber(psz, u32 / RT_US_1SEC, 10, 2, 0, RTSTR_F_ZEROPAD);
3818 *psz++ = '.';
3819 psz += RTStrFormatNumber(psz, u32 % RT_US_1SEC, 10, 6, 0, RTSTR_F_ZEROPAD);
3820 *psz++ = ' ';
3821 }
3822#define CCH_PREFIX_05 CCH_PREFIX_04 + (9+1+2+1+2+1+6+1)
3823
3824# if 0
3825 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_DATETIME)
3826 {
3827 char szDate[32];
3828 RTTIMESPEC Time;
3829 RTTimeSpecToString(RTTimeNow(&Time), szDate, sizeof(szDate));
3830 size_t cch = strlen(szDate);
3831 memcpy(psz, szDate, cch);
3832 psz += cch;
3833 *psz++ = ' ';
3834 }
3835# define CCH_PREFIX_06 CCH_PREFIX_05 + 32
3836# else
3837# define CCH_PREFIX_06 CCH_PREFIX_05 + 0
3838# endif
3839
3840 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_PID)
3841 {
3842#ifndef IN_RC
3843 RTPROCESS Process = RTProcSelf();
3844#else
3845 RTPROCESS Process = NIL_RTPROCESS;
3846#endif
3847 psz += RTStrFormatNumber(psz, Process, 16, sizeof(RTPROCESS) * 2, 0, RTSTR_F_ZEROPAD);
3848 *psz++ = ' ';
3849 }
3850#define CCH_PREFIX_07 CCH_PREFIX_06 + 9
3851
3852 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_TID)
3853 {
3854#ifndef IN_RC
3855 RTNATIVETHREAD Thread = RTThreadNativeSelf();
3856#else
3857 RTNATIVETHREAD Thread = NIL_RTNATIVETHREAD;
3858#endif
3859 psz += RTStrFormatNumber(psz, Thread, 16, sizeof(RTNATIVETHREAD) * 2, 0, RTSTR_F_ZEROPAD);
3860 *psz++ = ' ';
3861 }
3862#define CCH_PREFIX_08 CCH_PREFIX_07 + 17
3863
3864 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_THREAD)
3865 {
3866#ifdef IN_RING3
3867 const char *pszName = RTThreadSelfName();
3868#elif defined IN_RC
3869 const char *pszName = "EMT-RC";
3870#else
3871 const char *pszName = pLogger->pInt->szR0ThreadName[0] ? pLogger->pInt->szR0ThreadName : "R0";
3872#endif
3873 psz = rtLogStPNCpyPad(psz, pszName, 16, 8);
3874 }
3875#define CCH_PREFIX_09 CCH_PREFIX_08 + 17
3876
3877 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_CPUID)
3878 {
3879#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
3880 const uint8_t idCpu = ASMGetApicId();
3881#else
3882 const RTCPUID idCpu = RTMpCpuId();
3883#endif
3884 psz += RTStrFormatNumber(psz, idCpu, 16, sizeof(idCpu) * 2, 0, RTSTR_F_ZEROPAD);
3885 *psz++ = ' ';
3886 }
3887#define CCH_PREFIX_10 CCH_PREFIX_09 + 17
3888
3889#ifndef IN_RC
3890 if ( (pLogger->fFlags & RTLOGFLAGS_PREFIX_CUSTOM)
3891 && pLogger->pInt->pfnPrefix)
3892 {
3893 psz += pLogger->pInt->pfnPrefix(pLogger, psz, 31, pLogger->pInt->pvPrefixUserArg);
3894 *psz++ = ' '; /* +32 */
3895 }
3896#endif
3897#define CCH_PREFIX_11 CCH_PREFIX_10 + 32
3898
3899 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_LOCK_COUNTS)
3900 {
3901#ifdef IN_RING3 /** @todo implement these counters in ring-0 too? */
3902 RTTHREAD Thread = RTThreadSelf();
3903 if (Thread != NIL_RTTHREAD)
3904 {
3905 uint32_t cReadLocks = RTLockValidatorReadLockGetCount(Thread);
3906 uint32_t cWriteLocks = RTLockValidatorWriteLockGetCount(Thread) - g_cLoggerLockCount;
3907 cReadLocks = RT_MIN(0xfff, cReadLocks);
3908 cWriteLocks = RT_MIN(0xfff, cWriteLocks);
3909 psz += RTStrFormatNumber(psz, cReadLocks, 16, 1, 0, RTSTR_F_ZEROPAD);
3910 *psz++ = '/';
3911 psz += RTStrFormatNumber(psz, cWriteLocks, 16, 1, 0, RTSTR_F_ZEROPAD);
3912 }
3913 else
3914#endif
3915 {
3916 *psz++ = '?';
3917 *psz++ = '/';
3918 *psz++ = '?';
3919 }
3920 *psz++ = ' ';
3921 }
3922#define CCH_PREFIX_12 CCH_PREFIX_11 + 8
3923
3924 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_FLAG_NO)
3925 {
3926 psz += RTStrFormatNumber(psz, pArgs->fFlags, 16, 8, 0, RTSTR_F_ZEROPAD);
3927 *psz++ = ' ';
3928 }
3929#define CCH_PREFIX_13 CCH_PREFIX_12 + 9
3930
3931 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_FLAG)
3932 {
3933#ifdef IN_RING3
3934 const char *pszGroup = pArgs->iGroup != ~0U ? pLogger->pInt->papszGroups[pArgs->iGroup] : NULL;
3935#else
3936 const char *pszGroup = NULL;
3937#endif
3938 psz = rtLogStPNCpyPad(psz, pszGroup, 16, 8);
3939 }
3940#define CCH_PREFIX_14 CCH_PREFIX_13 + 17
3941
3942 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_GROUP_NO)
3943 {
3944 if (pArgs->iGroup != ~0U)
3945 {
3946 psz += RTStrFormatNumber(psz, pArgs->iGroup, 16, 3, 0, RTSTR_F_ZEROPAD);
3947 *psz++ = ' ';
3948 }
3949 else
3950 {
3951 memcpy(psz, "-1 ", sizeof("-1 ") - 1);
3952 psz += sizeof("-1 ") - 1;
3953 } /* +9 */
3954 }
3955#define CCH_PREFIX_15 CCH_PREFIX_14 + 9
3956
3957 if (pLogger->fFlags & RTLOGFLAGS_PREFIX_GROUP)
3958 {
3959 const unsigned fGrp = pLogger->afGroups[pArgs->iGroup != ~0U ? pArgs->iGroup : 0];
3960 const char *pszGroup;
3961 size_t cchGroup;
3962 switch (pArgs->fFlags & fGrp)
3963 {
3964 case 0: pszGroup = "--------"; cchGroup = sizeof("--------") - 1; break;
3965 case RTLOGGRPFLAGS_ENABLED: pszGroup = "enabled" ; cchGroup = sizeof("enabled" ) - 1; break;
3966 case RTLOGGRPFLAGS_LEVEL_1: pszGroup = "level 1" ; cchGroup = sizeof("level 1" ) - 1; break;
3967 case RTLOGGRPFLAGS_LEVEL_2: pszGroup = "level 2" ; cchGroup = sizeof("level 2" ) - 1; break;
3968 case RTLOGGRPFLAGS_LEVEL_3: pszGroup = "level 3" ; cchGroup = sizeof("level 3" ) - 1; break;
3969 case RTLOGGRPFLAGS_LEVEL_4: pszGroup = "level 4" ; cchGroup = sizeof("level 4" ) - 1; break;
3970 case RTLOGGRPFLAGS_LEVEL_5: pszGroup = "level 5" ; cchGroup = sizeof("level 5" ) - 1; break;
3971 case RTLOGGRPFLAGS_LEVEL_6: pszGroup = "level 6" ; cchGroup = sizeof("level 6" ) - 1; break;
3972 case RTLOGGRPFLAGS_LEVEL_7: pszGroup = "level 7" ; cchGroup = sizeof("level 7" ) - 1; break;
3973 case RTLOGGRPFLAGS_LEVEL_8: pszGroup = "level 8" ; cchGroup = sizeof("level 8" ) - 1; break;
3974 case RTLOGGRPFLAGS_LEVEL_9: pszGroup = "level 9" ; cchGroup = sizeof("level 9" ) - 1; break;
3975 case RTLOGGRPFLAGS_LEVEL_10: pszGroup = "level 10"; cchGroup = sizeof("level 10") - 1; break;
3976 case RTLOGGRPFLAGS_LEVEL_11: pszGroup = "level 11"; cchGroup = sizeof("level 11") - 1; break;
3977 case RTLOGGRPFLAGS_LEVEL_12: pszGroup = "level 12"; cchGroup = sizeof("level 12") - 1; break;
3978 case RTLOGGRPFLAGS_FLOW: pszGroup = "flow" ; cchGroup = sizeof("flow" ) - 1; break;
3979 case RTLOGGRPFLAGS_WARN: pszGroup = "warn" ; cchGroup = sizeof("warn" ) - 1; break;
3980 default: pszGroup = "????????"; cchGroup = sizeof("????????") - 1; break;
3981 }
3982 psz = rtLogStPNCpyPad2(psz, pszGroup, RT_MIN(cchGroup, 16), 8);
3983 }
3984#define CCH_PREFIX_16 CCH_PREFIX_15 + 17
3985
3986#define CCH_PREFIX ( CCH_PREFIX_16 )
3987 { AssertCompile(CCH_PREFIX < 256); }
3988
3989 /*
3990 * Done, figure what we've used and advance the buffer and free size.
3991 */
3992 cb = psz - &pLogger->achScratch[offScratch];
3993 AssertMsg(cb <= 223, ("%#zx (%zd) - fFlags=%#x\n", cb, cb, pLogger->fFlags));
3994 pLogger->offScratch = offScratch += (uint32_t)cb;
3995 cb = sizeof(pLogger->achScratch) - offScratch - 1;
3996 }
3997 else if (cb <= 0)
3998 {
3999 rtlogFlush(pLogger, true /*fNeedSpace*/);
4000 offScratch = pLogger->offScratch;
4001 cb = sizeof(pLogger->achScratch) - offScratch - 1;
4002 }
4003
4004#if defined(DEBUG) && defined(IN_RING3)
4005 /* sanity */
4006 if (offScratch >= sizeof(pLogger->achScratch))
4007 {
4008 fprintf(stderr, "offScratch >= sizeof(pLogger->achScratch) (%#x >= %#x)\n",
4009 offScratch, (unsigned)sizeof(pLogger->achScratch));
4010 AssertBreakpoint(); AssertBreakpoint();
4011 }
4012#endif
4013
4014 /* how much */
4015 if (cb > cbChars)
4016 cb = cbChars;
4017
4018 /* have newline? */
4019 pszNewLine = (const char *)memchr(pachChars, '\n', cb);
4020 if (pszNewLine)
4021 {
4022 if (pLogger->fFlags & RTLOGFLAGS_USECRLF)
4023 cb = pszNewLine - pachChars;
4024 else
4025 {
4026 cb = pszNewLine - pachChars + 1;
4027 *pfPendingPrefix = true;
4028 }
4029 }
4030
4031 /* copy */
4032 memcpy(&pLogger->achScratch[offScratch], pachChars, cb);
4033
4034 /* advance */
4035 pLogger->offScratch = offScratch += (uint32_t)cb;
4036 cbRet += cb;
4037 cbChars -= cb;
4038
4039 if ( pszNewLine
4040 && (pLogger->fFlags & RTLOGFLAGS_USECRLF)
4041 && offScratch + 2 < sizeof(pLogger->achScratch))
4042 {
4043 memcpy(&pLogger->achScratch[offScratch], "\r\n", 2);
4044 pLogger->offScratch = offScratch += 2;
4045 cbRet++;
4046 cbChars--;
4047 cb++;
4048 *pfPendingPrefix = true;
4049 }
4050
4051 /* done? */
4052 if (cbChars <= 0)
4053 return cbRet;
4054 pachChars += cb;
4055 }
4056
4057 /* won't ever get here! */
4058 }
4059 else
4060 {
4061 /*
4062 * Termination call.
4063 * There's always space for a terminator, and it's not counted.
4064 */
4065 pLogger->achScratch[pLogger->offScratch] = '\0';
4066 return 0;
4067 }
4068}
4069
4070
4071/**
4072 * Write to a logger instance (worker function).
4073 *
4074 * This function will check whether the instance, group and flags makes up a
4075 * logging kind which is currently enabled before writing anything to the log.
4076 *
4077 * @param pLogger Pointer to logger instance. Must be non-NULL.
4078 * @param fFlags The logging flags.
4079 * @param iGroup The group.
4080 * The value ~0U is reserved for compatibility with RTLogLogger[V] and is
4081 * only for internal usage!
4082 * @param pszFormat Format string.
4083 * @param args Format arguments.
4084 */
4085static void rtlogLoggerExVLocked(PRTLOGGER pLogger, unsigned fFlags, unsigned iGroup, const char *pszFormat, va_list args)
4086{
4087 /*
4088 * Format the message and perhaps flush it.
4089 */
4090 if (pLogger->fFlags & (RTLOGFLAGS_PREFIX_MASK | RTLOGFLAGS_USECRLF))
4091 {
4092 RTLOGOUTPUTPREFIXEDARGS OutputArgs;
4093 OutputArgs.pLogger = pLogger;
4094 OutputArgs.iGroup = iGroup;
4095 OutputArgs.fFlags = fFlags;
4096 RTLogFormatV(rtLogOutputPrefixed, &OutputArgs, pszFormat, args);
4097 }
4098 else
4099 RTLogFormatV(rtLogOutput, pLogger, pszFormat, args);
4100 if ( !(pLogger->fFlags & RTLOGFLAGS_BUFFERED)
4101 && pLogger->offScratch)
4102 rtlogFlush(pLogger, false /*fNeedSpace*/);
4103}
4104
4105
4106#ifndef IN_RC
4107/**
4108 * For calling rtlogLoggerExVLocked.
4109 *
4110 * @param pLogger The logger.
4111 * @param fFlags The logging flags.
4112 * @param iGroup The group.
4113 * The value ~0U is reserved for compatibility with RTLogLogger[V] and is
4114 * only for internal usage!
4115 * @param pszFormat Format string.
4116 * @param ... Format arguments.
4117 */
4118static void rtlogLoggerExFLocked(PRTLOGGER pLogger, unsigned fFlags, unsigned iGroup, const char *pszFormat, ...)
4119{
4120 va_list va;
4121 va_start(va, pszFormat);
4122 rtlogLoggerExVLocked(pLogger, fFlags, iGroup, pszFormat, va);
4123 va_end(va);
4124}
4125#endif /* !IN_RC */
4126
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