VirtualBox

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

Last change on this file since 82477 was 81072, checked in by vboxsync, 5 years ago

SUPDrv,IPRT,VMM: Support host APIC ID above 256 in GIP. (Only tested on 4 core intel.) [fix/hack] bugref:9501

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 22.0 KB
Line 
1/* $Id: tracebuf.cpp 81072 2019-09-30 10:30:01Z vboxsync $ */
2/** @file
3 * IPRT - Tracebuffer common functions.
4 */
5
6/*
7 * Copyright (C) 2011-2019 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 "internal/iprt.h"
32#include <iprt/trace.h>
33
34
35#include <iprt/assert.h>
36#include <iprt/asm.h>
37#include <iprt/errcore.h>
38#include <iprt/log.h>
39#ifndef IN_RC
40# include <iprt/mem.h>
41#endif
42#include <iprt/mp.h>
43#if defined(RT_ARCH_AMD64) || defined(RT_ARCH_X86)
44# include <iprt/asm-amd64-x86.h>
45#endif
46#include <iprt/path.h>
47#include <iprt/string.h>
48#include <iprt/time.h>
49
50#include "internal/magics.h"
51
52
53/*********************************************************************************************************************************
54* Structures and Typedefs *
55*********************************************************************************************************************************/
56/** Alignment used to place the trace buffer members, this should be a multiple
57 * of the cache line size if possible. (We should dynamically determine it.) */
58#define RTTRACEBUF_ALIGNMENT 64
59AssertCompile(RTTRACEBUF_ALIGNMENT >= sizeof(uint64_t) * 2);
60
61/** The maximum number of entries. */
62#define RTTRACEBUF_MAX_ENTRIES _64K
63/** The minimum number of entries. */
64#define RTTRACEBUF_MIN_ENTRIES 4
65/** The default number of entries. */
66#define RTTRACEBUF_DEF_ENTRIES 256
67
68/** The maximum entry size. */
69#define RTTRACEBUF_MAX_ENTRY_SIZE _1M
70/** The minimum entry size. */
71#define RTTRACEBUF_MIN_ENTRY_SIZE RTTRACEBUF_ALIGNMENT
72/** The default entry size. */
73#define RTTRACEBUF_DEF_ENTRY_SIZE 256
74AssertCompile(!(RTTRACEBUF_DEF_ENTRY_SIZE & (RTTRACEBUF_DEF_ENTRY_SIZE - 1)));
75
76/**
77 * The volatile trace buffer members.
78 */
79typedef struct RTTRACEBUFVOLATILE
80{
81 /** Reference counter. */
82 uint32_t volatile cRefs;
83 /** The next entry to make use of. */
84 uint32_t volatile iEntry;
85} RTTRACEBUFVOLATILE;
86/** Pointer to the volatile parts of a trace buffer. */
87typedef RTTRACEBUFVOLATILE *PRTTRACEBUFVOLATILE;
88
89
90/**
91 * Trace buffer entry.
92 */
93typedef struct RTTRACEBUFENTRY
94{
95 /** The nano second entry time stamp. */
96 uint64_t NanoTS;
97 /** The ID of the CPU the event was recorded. */
98 RTCPUID idCpu;
99 /** The message. */
100 char szMsg[RTTRACEBUF_ALIGNMENT - sizeof(uint64_t) - sizeof(RTCPUID)];
101} RTTRACEBUFENTRY;
102AssertCompile(sizeof(RTTRACEBUFENTRY) <= RTTRACEBUF_ALIGNMENT);
103/** Pointer to a trace buffer entry. */
104typedef RTTRACEBUFENTRY *PRTTRACEBUFENTRY;
105
106
107
108/**
109 * Trace buffer structure.
110 *
111 * @remarks This structure must be context agnostic, i.e. no pointers or
112 * other types that may differ between contexts (R3/R0/RC).
113 */
114typedef struct RTTRACEBUFINT
115{
116 /** Magic value (RTTRACEBUF_MAGIC). */
117 uint32_t u32Magic;
118 /** The entry size. */
119 uint32_t cbEntry;
120 /** The number of entries. */
121 uint32_t cEntries;
122 /** Flags (always zero for now). */
123 uint32_t fFlags;
124 /** The offset to the volatile members (RTTRACEBUFVOLATILE) (relative to
125 * the start of this structure). */
126 uint32_t offVolatile;
127 /** The offset to the entries (relative to the start of this structure). */
128 uint32_t offEntries;
129 /** Reserved entries. */
130 uint32_t au32Reserved[2];
131} RTTRACEBUFINT;
132/** Pointer to a const trace buffer. */
133typedef RTTRACEBUFINT const *PCRTTRACEBUFINT;
134
135
136/*********************************************************************************************************************************
137* Defined Constants And Macros *
138*********************************************************************************************************************************/
139/**
140 * Get the current CPU Id.
141 */
142#if defined(IN_RING0) \
143 || defined(RT_OS_WINDOWS) \
144 || (!defined(RT_ARCH_AMD64) && !defined(RT_ARCH_X86))
145# define RTTRACEBUF_CUR_CPU() RTMpCpuId()
146#else
147# define RTTRACEBUF_CUR_CPU() ASMGetApicId() /** @todo this isn't good enough for big boxes with lots of CPUs/cores. */
148#endif
149
150/** Calculates the address of the volatile trace buffer members. */
151#define RTTRACEBUF_TO_VOLATILE(a_pThis) ((PRTTRACEBUFVOLATILE)((uint8_t *)(a_pThis) + (a_pThis)->offVolatile))
152
153/** Calculates the address of a trace buffer entry. */
154#define RTTRACEBUF_TO_ENTRY(a_pThis, a_iEntry) \
155 ((PRTTRACEBUFENTRY)( (uint8_t *)(a_pThis) + (a_pThis)->offEntries + (a_iEntry) * (a_pThis)->cbEntry ))
156
157/** Validates a trace buffer handle and returns rc if not valid. */
158#define RTTRACEBUF_VALID_RETURN_RC(a_pThis, a_rc) \
159 do { \
160 AssertPtrReturn((a_pThis), (a_rc)); \
161 AssertReturn((a_pThis)->u32Magic == RTTRACEBUF_MAGIC, (a_rc)); \
162 AssertReturn((a_pThis)->offVolatile < RTTRACEBUF_ALIGNMENT * 2, (a_rc)); \
163 AssertReturn(RTTRACEBUF_TO_VOLATILE(a_pThis)->cRefs > 0, (a_rc)); \
164 } while (0)
165
166/**
167 * Resolves and validates a trace buffer handle and returns rc if not valid.
168 *
169 * @param a_hTraceBuf The trace buffer handle passed by the user.
170 * @param a_pThis Where to store the trace buffer pointer.
171 */
172#define RTTRACEBUF_RESOLVE_VALIDATE_RETAIN_RETURN(a_hTraceBuf, a_pThis) \
173 do { \
174 uint32_t cRefs; \
175 if ((a_hTraceBuf) == RTTRACEBUF_DEFAULT) \
176 { \
177 (a_pThis) = RTTraceGetDefaultBuf(); \
178 if (!RT_VALID_PTR(a_pThis)) \
179 return VERR_NOT_FOUND; \
180 } \
181 else \
182 { \
183 (a_pThis) = (a_hTraceBuf); \
184 AssertPtrReturn((a_pThis), VERR_INVALID_HANDLE); \
185 } \
186 AssertReturn((a_pThis)->u32Magic == RTTRACEBUF_MAGIC, VERR_INVALID_HANDLE); \
187 AssertReturn((a_pThis)->offVolatile < RTTRACEBUF_ALIGNMENT * 2, VERR_INVALID_HANDLE); \
188 \
189 cRefs = ASMAtomicIncU32(&RTTRACEBUF_TO_VOLATILE(a_pThis)->cRefs); \
190 if (RT_UNLIKELY(cRefs < 1 || cRefs >= _1M)) \
191 { \
192 ASMAtomicDecU32(&RTTRACEBUF_TO_VOLATILE(a_pThis)->cRefs); \
193 AssertFailedReturn(VERR_INVALID_HANDLE); \
194 } \
195 } while (0)
196
197
198/**
199 * Drops a trace buffer reference.
200 *
201 * @param a_pThis Pointer to the trace buffer.
202 */
203#define RTTRACEBUF_DROP_REFERENCE(a_pThis) \
204 do { \
205 uint32_t cRefs = ASMAtomicDecU32(&RTTRACEBUF_TO_VOLATILE(a_pThis)->cRefs); \
206 if (!cRefs) \
207 rtTraceBufDestroy((RTTRACEBUFINT *)a_pThis); \
208 } while (0)
209
210
211/**
212 * The prologue code for a RTTraceAddSomething function.
213 *
214 * Resolves a trace buffer handle, grabs a reference to it and allocates the
215 * next entry. Return with an appropriate error status on failure.
216 *
217 * @param a_hTraceBuf The trace buffer handle passed by the user.
218 *
219 * @remarks This is kind of ugly, sorry.
220 */
221#define RTTRACEBUF_ADD_PROLOGUE(a_hTraceBuf) \
222 int rc; \
223 uint32_t cRefs; \
224 uint32_t iEntry; \
225 PCRTTRACEBUFINT pThis; \
226 PRTTRACEBUFVOLATILE pVolatile; \
227 PRTTRACEBUFENTRY pEntry; \
228 char *pszBuf; \
229 size_t cchBuf; \
230 \
231 /* Resolve and validate the handle. */ \
232 if ((a_hTraceBuf) == RTTRACEBUF_DEFAULT) \
233 { \
234 pThis = RTTraceGetDefaultBuf(); \
235 if (!RT_VALID_PTR(pThis)) \
236 return VERR_NOT_FOUND; \
237 } \
238 else if ((a_hTraceBuf) != NIL_RTTRACEBUF) \
239 { \
240 pThis = (a_hTraceBuf); \
241 AssertPtrReturn(pThis, VERR_INVALID_HANDLE); \
242 } \
243 else \
244 return VERR_INVALID_HANDLE; \
245 \
246 AssertReturn(pThis->u32Magic == RTTRACEBUF_MAGIC, VERR_INVALID_HANDLE); \
247 if (pThis->fFlags & RTTRACEBUF_FLAGS_DISABLED) \
248 return VINF_SUCCESS; \
249 AssertReturn(pThis->offVolatile < RTTRACEBUF_ALIGNMENT * 2, VERR_INVALID_HANDLE); \
250 pVolatile = RTTRACEBUF_TO_VOLATILE(pThis); \
251 \
252 /* Grab a reference. */ \
253 cRefs = ASMAtomicIncU32(&pVolatile->cRefs); \
254 if (RT_UNLIKELY(cRefs < 1 || cRefs >= _1M)) \
255 { \
256 ASMAtomicDecU32(&pVolatile->cRefs); \
257 AssertFailedReturn(VERR_INVALID_HANDLE); \
258 } \
259 \
260 /* Grab the next entry and set the time stamp. */ \
261 iEntry = ASMAtomicIncU32(&pVolatile->iEntry) - 1; \
262 iEntry %= pThis->cEntries; \
263 pEntry = RTTRACEBUF_TO_ENTRY(pThis, iEntry); \
264 pEntry->NanoTS = RTTimeNanoTS(); \
265 pEntry->idCpu = RTTRACEBUF_CUR_CPU(); \
266 pszBuf = &pEntry->szMsg[0]; \
267 *pszBuf = '\0'; \
268 cchBuf = pThis->cbEntry - RT_UOFFSETOF(RTTRACEBUFENTRY, szMsg) - 1; \
269 rc = VINF_SUCCESS
270
271
272/**
273 * Used by a RTTraceAddPosSomething to store the source position in the entry
274 * prior to adding the actual trace message text.
275 *
276 * Both pszBuf and cchBuf will be adjusted such that pszBuf points and the zero
277 * terminator after the source position part.
278 */
279#define RTTRACEBUF_ADD_STORE_SRC_POS() \
280 do { \
281 /* file(line): - no path */ \
282 size_t cchPos = RTStrPrintf(pszBuf, cchBuf, "%s(%d): ", RTPathFilename(pszFile), iLine); \
283 pszBuf += cchPos; \
284 cchBuf -= cchPos; \
285 NOREF(pszFunction); \
286 } while (0)
287
288
289/**
290 * The epilogue code for a RTTraceAddSomething function.
291 *
292 * This will release the trace buffer reference.
293 */
294#define RTTRACEBUF_ADD_EPILOGUE() \
295 cRefs = ASMAtomicDecU32(&pVolatile->cRefs); \
296 if (!cRefs) \
297 rtTraceBufDestroy((RTTRACEBUFINT *)pThis); \
298 return rc
299
300
301#ifndef IN_RC /* Drop this in RC context (too lazy to split the file). */
302
303RTDECL(int) RTTraceBufCreate(PRTTRACEBUF phTraceBuf, uint32_t cEntries, uint32_t cbEntry, uint32_t fFlags)
304{
305 AssertPtrReturn(phTraceBuf, VERR_INVALID_POINTER);
306 AssertReturn(!(fFlags & ~(RTTRACEBUF_FLAGS_MASK & ~ RTTRACEBUF_FLAGS_FREE_ME)), VERR_INVALID_PARAMETER);
307 AssertMsgReturn(cbEntry <= RTTRACEBUF_MAX_ENTRIES, ("%#x\n", cbEntry), VERR_OUT_OF_RANGE);
308 AssertMsgReturn(cEntries <= RTTRACEBUF_MAX_ENTRY_SIZE, ("%#x\n", cEntries), VERR_OUT_OF_RANGE);
309
310 /*
311 * Apply default and alignment adjustments.
312 */
313 if (!cbEntry)
314 cbEntry = RTTRACEBUF_DEF_ENTRY_SIZE;
315 else
316 cbEntry = RT_ALIGN_32(cbEntry, RTTRACEBUF_ALIGNMENT);
317
318 if (!cEntries)
319 cEntries = RTTRACEBUF_DEF_ENTRIES;
320 else if (cEntries < RTTRACEBUF_MIN_ENTRIES)
321 cEntries = RTTRACEBUF_MIN_ENTRIES;
322
323 /*
324 * Calculate the required buffer size, allocte it and hand it on to the
325 * carver API.
326 */
327 size_t cbBlock = cbEntry * cEntries
328 + RT_ALIGN_Z(sizeof(RTTRACEBUFINT), RTTRACEBUF_ALIGNMENT)
329 + RT_ALIGN_Z(sizeof(RTTRACEBUFVOLATILE), RTTRACEBUF_ALIGNMENT);
330 void *pvBlock = RTMemAlloc(cbBlock);
331 if (!((uintptr_t)pvBlock & (RTTRACEBUF_ALIGNMENT - 1)))
332 {
333 RTMemFree(pvBlock);
334 cbBlock += RTTRACEBUF_ALIGNMENT - 1;
335 pvBlock = RTMemAlloc(cbBlock);
336 }
337 int rc;
338 if (pvBlock)
339 {
340 rc = RTTraceBufCarve(phTraceBuf, cEntries, cbEntry, fFlags, pvBlock, &cbBlock);
341 if (RT_FAILURE(rc))
342 RTMemFree(pvBlock);
343 }
344 else
345 rc = VERR_NO_MEMORY;
346 return rc;
347}
348
349
350RTDECL(int) RTTraceBufCarve(PRTTRACEBUF phTraceBuf, uint32_t cEntries, uint32_t cbEntry, uint32_t fFlags,
351 void *pvBlock, size_t *pcbBlock)
352{
353 AssertPtrReturn(phTraceBuf, VERR_INVALID_POINTER);
354 AssertReturn(!(fFlags & ~RTTRACEBUF_FLAGS_MASK), VERR_INVALID_PARAMETER);
355 AssertMsgReturn(cbEntry <= RTTRACEBUF_MAX_ENTRIES, ("%#x\n", cbEntry), VERR_OUT_OF_RANGE);
356 AssertMsgReturn(cEntries <= RTTRACEBUF_MAX_ENTRY_SIZE, ("%#x\n", cEntries), VERR_OUT_OF_RANGE);
357 AssertPtrReturn(pcbBlock, VERR_INVALID_POINTER);
358 size_t const cbBlock = *pcbBlock;
359 AssertReturn(RT_VALID_PTR(pvBlock) || !cbBlock, VERR_INVALID_POINTER);
360
361 /*
362 * Apply defaults, align sizes and check against available buffer space.
363 * This code can be made a bit more clever, if someone feels like it.
364 */
365 size_t const cbHdr = RT_ALIGN_Z(sizeof(RTTRACEBUFINT), RTTRACEBUF_ALIGNMENT)
366 + RT_ALIGN_Z(sizeof(RTTRACEBUFVOLATILE), RTTRACEBUF_ALIGNMENT);
367 size_t const cbEntryBuf = cbBlock > cbHdr ? cbBlock - cbHdr : 0;
368 if (cbEntry)
369 cbEntry = RT_ALIGN_32(cbEntry, RTTRACEBUF_ALIGNMENT);
370 else
371 {
372 if (!cbEntryBuf)
373 {
374 cbEntry = RTTRACEBUF_DEF_ENTRY_SIZE;
375 cEntries = RTTRACEBUF_DEF_ENTRIES;
376 }
377 else if (cEntries)
378 {
379 size_t cbEntryZ = cbBlock / cEntries;
380 cbEntryZ &= ~(RTTRACEBUF_ALIGNMENT - 1);
381 if (cbEntryZ > RTTRACEBUF_MAX_ENTRIES)
382 cbEntryZ = RTTRACEBUF_MAX_ENTRIES;
383 cbEntry = (uint32_t)cbEntryZ;
384 }
385 else if (cbBlock >= RT_ALIGN_32(512, RTTRACEBUF_ALIGNMENT) * 256)
386 cbEntry = RT_ALIGN_32(512, RTTRACEBUF_ALIGNMENT);
387 else if (cbBlock >= RT_ALIGN_32(256, RTTRACEBUF_ALIGNMENT) * 64)
388 cbEntry = RT_ALIGN_32(256, RTTRACEBUF_ALIGNMENT);
389 else if (cbBlock >= RT_ALIGN_32(128, RTTRACEBUF_ALIGNMENT) * 32)
390 cbEntry = RT_ALIGN_32(128, RTTRACEBUF_ALIGNMENT);
391 else
392 cbEntry = sizeof(RTTRACEBUFENTRY);
393 }
394 Assert(RT_ALIGN_32(cbEntry, RTTRACEBUF_ALIGNMENT) == cbEntry);
395
396 if (!cEntries)
397 {
398 size_t cEntriesZ = cbEntryBuf / cbEntry;
399 if (cEntriesZ > RTTRACEBUF_MAX_ENTRIES)
400 cEntriesZ = RTTRACEBUF_MAX_ENTRIES;
401 cEntries = (uint32_t)cEntriesZ;
402 }
403 if (cEntries < RTTRACEBUF_MIN_ENTRIES)
404 cEntries = RTTRACEBUF_MIN_ENTRIES;
405
406 uint32_t offVolatile = RTTRACEBUF_ALIGNMENT - ((uintptr_t)pvBlock & (RTTRACEBUF_ALIGNMENT - 1));
407 if (offVolatile < sizeof(RTTRACEBUFINT))
408 offVolatile += RTTRACEBUF_ALIGNMENT;
409 size_t cbReqBlock = offVolatile
410 + RT_ALIGN_Z(sizeof(RTTRACEBUFVOLATILE), RTTRACEBUF_ALIGNMENT)
411 + cbEntry * cEntries;
412 if (*pcbBlock < cbReqBlock)
413 {
414 *pcbBlock = cbReqBlock;
415 return VERR_BUFFER_OVERFLOW;
416 }
417
418 /*
419 * Do the carving.
420 */
421 memset(pvBlock, 0, cbBlock);
422
423 RTTRACEBUFINT *pThis = (RTTRACEBUFINT *)pvBlock;
424 pThis->u32Magic = RTTRACEBUF_MAGIC;
425 pThis->cbEntry = cbEntry;
426 pThis->cEntries = cEntries;
427 pThis->fFlags = fFlags;
428 pThis->offVolatile = offVolatile;
429 pThis->offEntries = offVolatile + RT_ALIGN_Z(sizeof(RTTRACEBUFVOLATILE), RTTRACEBUF_ALIGNMENT);
430
431 PRTTRACEBUFVOLATILE pVolatile = (PRTTRACEBUFVOLATILE)((uint8_t *)pThis + offVolatile);
432 pVolatile->cRefs = 1;
433 pVolatile->iEntry = 0;
434
435 *pcbBlock = cbBlock - cbReqBlock;
436 *phTraceBuf = pThis;
437 return VINF_SUCCESS;
438}
439
440#endif /* !IN_RC */
441
442
443/**
444 * Destructor.
445 *
446 * @param pThis The trace buffer to destroy.
447 */
448static void rtTraceBufDestroy(RTTRACEBUFINT *pThis)
449{
450 AssertReturnVoid(ASMAtomicCmpXchgU32(&pThis->u32Magic, RTTRACEBUF_MAGIC_DEAD, RTTRACEBUF_MAGIC));
451 if (pThis->fFlags & RTTRACEBUF_FLAGS_FREE_ME)
452 {
453#ifdef IN_RC
454 AssertReleaseFailed();
455#else
456 RTMemFree(pThis);
457#endif
458 }
459}
460
461
462RTDECL(uint32_t) RTTraceBufRetain(RTTRACEBUF hTraceBuf)
463{
464 PCRTTRACEBUFINT pThis = hTraceBuf;
465 RTTRACEBUF_VALID_RETURN_RC(pThis, UINT32_MAX);
466 return ASMAtomicIncU32(&RTTRACEBUF_TO_VOLATILE(pThis)->cRefs);
467}
468
469
470RTDECL(uint32_t) RTTraceBufRelease(RTTRACEBUF hTraceBuf)
471{
472 if (hTraceBuf == NIL_RTTRACEBUF)
473 return 0;
474
475 PCRTTRACEBUFINT pThis = hTraceBuf;
476 RTTRACEBUF_VALID_RETURN_RC(pThis, UINT32_MAX);
477
478 uint32_t cRefs = ASMAtomicDecU32(&RTTRACEBUF_TO_VOLATILE(pThis)->cRefs);
479 if (!cRefs)
480 rtTraceBufDestroy((RTTRACEBUFINT *)pThis);
481 return cRefs;
482}
483
484
485RTDECL(int) RTTraceBufAddMsg(RTTRACEBUF hTraceBuf, const char *pszMsg)
486{
487 RTTRACEBUF_ADD_PROLOGUE(hTraceBuf);
488 RTStrCopy(pszBuf, cchBuf, pszMsg);
489 RTTRACEBUF_ADD_EPILOGUE();
490}
491
492
493RTDECL(int) RTTraceBufAddMsgEx( RTTRACEBUF hTraceBuf, const char *pszMsg, size_t cbMaxMsg)
494{
495 RTTRACEBUF_ADD_PROLOGUE(hTraceBuf);
496 RTStrCopyEx(pszBuf, cchBuf, pszMsg, cbMaxMsg);
497 RTTRACEBUF_ADD_EPILOGUE();
498}
499
500
501RTDECL(int) RTTraceBufAddMsgF(RTTRACEBUF hTraceBuf, const char *pszMsgFmt, ...)
502{
503 int rc;
504 va_list va;
505 va_start(va, pszMsgFmt);
506 rc = RTTraceBufAddMsgV(hTraceBuf, pszMsgFmt, va);
507 va_end(va);
508 return rc;
509}
510
511
512RTDECL(int) RTTraceBufAddMsgV(RTTRACEBUF hTraceBuf, const char *pszMsgFmt, va_list va)
513{
514 RTTRACEBUF_ADD_PROLOGUE(hTraceBuf);
515 RTStrPrintfV(pszBuf, cchBuf, pszMsgFmt, va);
516 RTTRACEBUF_ADD_EPILOGUE();
517}
518
519
520RTDECL(int) RTTraceBufAddPos(RTTRACEBUF hTraceBuf, RT_SRC_POS_DECL)
521{
522 RTTRACEBUF_ADD_PROLOGUE(hTraceBuf);
523 RTTRACEBUF_ADD_STORE_SRC_POS();
524 RTTRACEBUF_ADD_EPILOGUE();
525}
526
527
528RTDECL(int) RTTraceBufAddPosMsg(RTTRACEBUF hTraceBuf, RT_SRC_POS_DECL, const char *pszMsg)
529{
530 RTTRACEBUF_ADD_PROLOGUE(hTraceBuf);
531 RTTRACEBUF_ADD_STORE_SRC_POS();
532 RTStrCopy(pszBuf, cchBuf, pszMsg);
533 RTTRACEBUF_ADD_EPILOGUE();
534}
535
536
537RTDECL(int) RTTraceBufAddPosMsgEx(RTTRACEBUF hTraceBuf, RT_SRC_POS_DECL, const char *pszMsg, size_t cbMaxMsg)
538{
539 RTTRACEBUF_ADD_PROLOGUE(hTraceBuf);
540 RTTRACEBUF_ADD_STORE_SRC_POS();
541 RTStrCopyEx(pszBuf, cchBuf, pszMsg, cbMaxMsg);
542 RTTRACEBUF_ADD_EPILOGUE();
543}
544
545
546RTDECL(int) RTTraceBufAddPosMsgF(RTTRACEBUF hTraceBuf, RT_SRC_POS_DECL, const char *pszMsgFmt, ...)
547{
548 int rc;
549 va_list va;
550 va_start(va, pszMsgFmt);
551 rc = RTTraceBufAddPosMsgV(hTraceBuf, RT_SRC_POS_ARGS, pszMsgFmt, va);
552 va_end(va);
553 return rc;
554}
555
556
557RTDECL(int) RTTraceBufAddPosMsgV(RTTRACEBUF hTraceBuf, RT_SRC_POS_DECL, const char *pszMsgFmt, va_list va)
558{
559 RTTRACEBUF_ADD_PROLOGUE(hTraceBuf);
560 RTTRACEBUF_ADD_STORE_SRC_POS();
561 RTStrPrintfV(pszBuf, cchBuf, pszMsgFmt, va);
562 RTTRACEBUF_ADD_EPILOGUE();
563}
564
565
566RTDECL(int) RTTraceBufEnumEntries(RTTRACEBUF hTraceBuf, PFNRTTRACEBUFCALLBACK pfnCallback, void *pvUser)
567{
568 int rc = VINF_SUCCESS;
569 uint32_t iBase;
570 uint32_t cLeft;
571 PCRTTRACEBUFINT pThis;
572 RTTRACEBUF_RESOLVE_VALIDATE_RETAIN_RETURN(hTraceBuf, pThis);
573
574 iBase = ASMAtomicReadU32(&RTTRACEBUF_TO_VOLATILE(pThis)->iEntry);
575 cLeft = pThis->cEntries;
576 while (cLeft--)
577 {
578 PRTTRACEBUFENTRY pEntry;
579
580 iBase %= pThis->cEntries;
581 pEntry = RTTRACEBUF_TO_ENTRY(pThis, iBase);
582 if (pEntry->NanoTS)
583 {
584 rc = pfnCallback((RTTRACEBUF)pThis, cLeft, pEntry->NanoTS, pEntry->idCpu, pEntry->szMsg, pvUser);
585 if (rc != VINF_SUCCESS)
586 break;
587 }
588
589 /* next */
590 iBase += 1;
591 }
592
593 RTTRACEBUF_DROP_REFERENCE(pThis);
594 return rc;
595}
596
597
598RTDECL(uint32_t) RTTraceBufGetEntrySize(RTTRACEBUF hTraceBuf)
599{
600 PCRTTRACEBUFINT pThis = hTraceBuf;
601 RTTRACEBUF_VALID_RETURN_RC(pThis, 0);
602 return pThis->cbEntry;
603}
604
605
606RTDECL(uint32_t) RTTraceBufGetEntryCount(RTTRACEBUF hTraceBuf)
607{
608 PCRTTRACEBUFINT pThis = hTraceBuf;
609 RTTRACEBUF_VALID_RETURN_RC(pThis, 0);
610 return pThis->cEntries;
611}
612
613
614RTDECL(bool) RTTraceBufDisable(RTTRACEBUF hTraceBuf)
615{
616 PCRTTRACEBUFINT pThis = hTraceBuf;
617 RTTRACEBUF_VALID_RETURN_RC(pThis, false);
618 return !ASMAtomicBitTestAndSet((void volatile *)&pThis->fFlags, RTTRACEBUF_FLAGS_DISABLED_BIT);
619}
620
621
622RTDECL(bool) RTTraceBufEnable(RTTRACEBUF hTraceBuf)
623{
624 PCRTTRACEBUFINT pThis = hTraceBuf;
625 RTTRACEBUF_VALID_RETURN_RC(pThis, false);
626 return !ASMAtomicBitTestAndClear((void volatile *)&pThis->fFlags, RTTRACEBUF_FLAGS_DISABLED_BIT);
627}
628
629
630/*
631 *
632 * Move the following to a separate file, consider using the enumerator.
633 *
634 */
635
636RTDECL(int) RTTraceBufDumpToLog(RTTRACEBUF hTraceBuf)
637{
638 uint32_t iBase;
639 uint32_t cLeft;
640 PCRTTRACEBUFINT pThis;
641 RTTRACEBUF_RESOLVE_VALIDATE_RETAIN_RETURN(hTraceBuf, pThis);
642
643 iBase = ASMAtomicReadU32(&RTTRACEBUF_TO_VOLATILE(pThis)->iEntry);
644 cLeft = pThis->cEntries;
645 while (cLeft--)
646 {
647 PRTTRACEBUFENTRY pEntry;
648
649 iBase %= pThis->cEntries;
650 pEntry = RTTRACEBUF_TO_ENTRY(pThis, iBase);
651 if (pEntry->NanoTS)
652 RTLogPrintf("%04u/%'llu/%02x: %s\n", cLeft, pEntry->NanoTS, pEntry->idCpu, pEntry->szMsg);
653
654 /* next */
655 iBase += 1;
656 }
657
658 RTTRACEBUF_DROP_REFERENCE(pThis);
659 return VINF_SUCCESS;
660}
661
662
663RTDECL(int) RTTraceBufDumpToAssert(RTTRACEBUF hTraceBuf)
664{
665 uint32_t iBase;
666 uint32_t cLeft;
667 PCRTTRACEBUFINT pThis;
668 RTTRACEBUF_RESOLVE_VALIDATE_RETAIN_RETURN(hTraceBuf, pThis);
669
670 iBase = ASMAtomicReadU32(&RTTRACEBUF_TO_VOLATILE(pThis)->iEntry);
671 cLeft = pThis->cEntries;
672 while (cLeft--)
673 {
674 PRTTRACEBUFENTRY pEntry;
675
676 iBase %= pThis->cEntries;
677 pEntry = RTTRACEBUF_TO_ENTRY(pThis, iBase);
678 if (pEntry->NanoTS)
679 RTAssertMsg2AddWeak("%u/%'llu/%02x: %s\n", cLeft, pEntry->NanoTS, pEntry->idCpu, pEntry->szMsg);
680
681 /* next */
682 iBase += 1;
683 }
684
685 RTTRACEBUF_DROP_REFERENCE(pThis);
686 return VINF_SUCCESS;
687}
688
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