VirtualBox

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

Last change on this file since 83291 was 82968, checked in by vboxsync, 5 years ago

Copyright year updates by scm.

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