VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/string/strformatrt.cpp@ 7389

Last change on this file since 7389 was 5999, checked in by vboxsync, 17 years ago

The Giant CDDL Dual-License Header Change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 35.2 KB
Line 
1/* $Id: strformatrt.cpp 5999 2007-12-07 15:05:06Z vboxsync $ */
2/** @file
3 * innotek Portable Runtime - IPRT String Formatter Extensions.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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/** @page pg_rt_str_format_rt The IPRT String Format Extensions
28 *
29 * The string formatter supports most of the non-float format types and flags.
30 * See RTStrFormatV() for the full tail there. In addition we've added a number
31 * of iprt specific format types for the iprt typedefs and other useful stuff.
32 * Note that several of these are similar to \%p and doesn't care much if you try
33 * add formating flags/width/precision.
34 *
35 *
36 * Group 1, the basic runtime typedefs (excluding those which obviously are pointer).
37 * - \%RTbool - Takes a bool value and prints 'true', 'false', or '!%d!'.
38 * - \%RTfile - Takes a #RTFILE value.
39 * - \%RTfmode - Takes a #RTFMODE value.
40 * - \%RTfoff - Takes a #RTFOFF value.
41 * - \%RTfp16 - Takes a #RTFAR16 value.
42 * - \%RTfp32 - Takes a #RTFAR32 value.
43 * - \%RTfp64 - Takes a #RTFAR64 value.
44 * - \%RTgid - Takes a #RTGID value.
45 * - \%RTino - Takes a #RTINODE value.
46 * - \%RTint - Takes a #RTINT value.
47 * - \%RTiop - Takes a #RTIOPORT value.
48 * - \%RTldrm - Takes a #RTLDRMOD value.
49 * - \%RTnthrd - Takes a #RTNATIVETHREAD value.
50 * - \%RTproc - Takes a #RTPROCESS value.
51 * - \%RTptr - Takes a #RTINTPTR or #RTUINTPTR value (but not void *).
52 * - \%RTreg - Takes a #RTUINTREG value.
53 * - \%RTsel - Takes a #RTSEL value.
54 * - \%RTsem - Takes a #RTSEMEVENT, #RTSEMEVENTMULTI, #RTSEMMUTEX, #RTSEMFASTMUTEX, or #RTSEMRW value.
55 * - \%RTsock - Takes a #RTSOCKET value.
56 * - \%RTthrd - Takes a #RTTHREAD value.
57 * - \%RTuid - Takes a #RTUID value.
58 * - \%RTuint - Takes a #RTUINT value.
59 * - \%RTunicp - Takes a #RTUNICP value.
60 * - \%RTutf16 - Takes a #RTUTF16 value.
61 * - \%RTuuid - Takes a #PCRTUUID and will print the UUID as a string.
62 * - \%RTxuint - Takes a #RTUINT or #RTINT value, formatting it as hex.
63 * - \%RGi - Takes a #RTGCINT value.
64 * - \%RGp - Takes a #RTGCPHYS value.
65 * - \%RGr - Takes a #RTGCUINTREG value.
66 * - \%RGu - Takes a #RTGCUINT value.
67 * - \%RGv - Takes a #RTGCPTR, #RTGCINTPTR or #RTGCUINTPTR value.
68 * - \%RGx - Takes a #RTGCUINT or #RTGCINT value, formatting it as hex.
69 * - \%RHi - Takes a #RTHCINT value.
70 * - \%RHp - Takes a #RTHCPHYS value.
71 * - \%RHr - Takes a #RTHCUINTREG value.
72 * - \%RHu - Takes a #RTHCUINT value.
73 * - \%RHv - Takes a #RTHCPTR, #RTHCINTPTR or #RTHCUINTPTR value.
74 * - \%RHx - Takes a #RTHCUINT or #RTHCINT value, formatting it as hex.
75 * - \%RCi - Takes a #RTCCINT value.
76 * - \%RCp - Takes a #RTCCPHYS value.
77 * - \%RCr - Takes a #RTCCUINTREG value.
78 * - \%RCu - Takes a #RTUINT value.
79 * - \%RCv - Takes a #uintptr_t, #intptr_t, void * value.
80 * - \%RCx - Takes a #RTUINT or #RTINT value, formatting it as hex.
81 *
82 *
83 * Group 2, the generic integer types which are prefered over relying on what
84 * bit-count a 'long', 'short', or 'long long' has on a platform. This are
85 * highly prefered for the [u]intXX_t kind of types.
86 * - \%RI[8|16|32|64] - Signed integer value of the specifed bit count.
87 * - \%RU[8|16|32|64] - Unsigned integer value of the specifed bit count.
88 * - \%RX[8|16|32|64] - Hexadecimal integer value of the specifed bit count.
89 *
90 *
91 * Group 3, hex dumpers and other complex stuff which requires more than simple formatting.
92 * - \%Rhxd - Takes a pointer to the memory which is to be dumped in typical
93 * hex format. Use the width to specify the length, and the precision to
94 * set the number of bytes per line. Default width and precision is 16.
95 * - \%Rhxs - Takes a pointer to the memory to be displayed as a hex string,
96 * i.e. a series of space separated bytes formatted as two digit hex value.
97 * Use the width to specify the length. Default length is 16 bytes.
98 * - \%Rrc - Takes an integer iprt status code as argument. Will insert the
99 * status code define corresponding to the iprt status code.
100 * - \%Rrs - Takes an integer iprt status code as argument. Will insert the
101 * short description of the specified status code.
102 * - \%Rrf - Takes an integer iprt status code as argument. Will insert the
103 * full description of the specified status code.
104 * - \%Rra - Takes an integer iprt status code as argument. Will insert the
105 * status code define + full description.
106 * - \%Rt - Current thread (RTThreadSelf()), no arguments.
107 *
108 * - \%Rwc - Takes a long Windows error code as argument. Will insert the status
109 * code define corresponding to the Windows error code.
110 * - \%Rwf - Takes a long Windows error code as argument. Will insert the
111 * full description of the specified status code.
112 * - \%Rwa - Takes a long Windows error code as argument. Will insert the
113 * error code define + full description.
114 *
115 * On other platforms, \%Rw? simply prints the argument in a form of 0xXXXXXXXX.
116 *
117 * @todo (r=dmik) Add a cross-platform \%Rcomr? that will fall back to \%Rw? on
118 * Win32 platforms and will interpret XPCOM result codes on all other.
119 *
120 *
121 * Group 4, structure dumpers.
122 *
123 * - \%RDtimespec - Takes a PCRTTIMESPEC.
124 *
125 *
126 */
127
128/*******************************************************************************
129* Header Files *
130*******************************************************************************/
131#define LOG_GROUP RTLOGGROUP_STRING
132#include <iprt/log.h>
133#include <iprt/string.h>
134#include <iprt/assert.h>
135#include <iprt/string.h>
136#include <iprt/stdarg.h>
137#ifdef IN_RING3
138# include <iprt/thread.h>
139# include <iprt/err.h>
140#endif
141#include <iprt/time.h>
142#include "internal/string.h"
143
144
145
146/**
147 * Callback to format iprt formatting extentions.
148 * See @ref pg_rt_str_format_rt for a reference on the format types.
149 *
150 * @returns The number of bytes formatted.
151 * @param pfnOutput Pointer to output function.
152 * @param pvArgOutput Argument for the output function.
153 * @param ppszFormat Pointer to the format string pointer. Advance this till the char
154 * after the format specifier.
155 * @param pArgs Pointer to the argument list. Use this to fetch the arguments.
156 * @param cchWidth Format Width. -1 if not specified.
157 * @param cchPrecision Format Precision. -1 if not specified.
158 * @param fFlags Flags (RTSTR_NTFS_*).
159 * @param chArgSize The argument size specifier, 'l' or 'L'.
160 */
161size_t rtstrFormatRt(PFNRTSTROUTPUT pfnOutput, void *pvArgOutput, const char **ppszFormat, va_list *pArgs, int cchWidth, int cchPrecision, unsigned fFlags, char chArgSize)
162{
163 const char *pszFormatOrg = *ppszFormat;
164 char ch = *(*ppszFormat)++;
165 if (ch == 'R')
166 {
167 ch = *(*ppszFormat)++;
168 switch (ch)
169 {
170 /*
171 * Groups 1 and 2.
172 */
173 case 'T':
174 case 'G':
175 case 'H':
176 case 'C':
177 case 'I':
178 case 'X':
179 case 'U':
180 {
181 /*
182 * Interpret the type.
183 */
184 typedef enum { RTSF_INT, RTSF_INTW, RTSF_FP16, RTSF_FP32, RTSF_FP64, RTSF_UUID, RTSF_BOOL } RTSF;
185 static const struct
186 {
187 uint8_t cch; /**< the length of the string. */
188 char sz[10]; /**< the part following 'R'. */
189 uint8_t cb; /**< the size of the type. */
190 uint8_t u8Base; /**< the size of the type. */
191 RTSF enmFormat; /**< The way to format it. */
192 uint16_t fFlags; /**< additional RTSTR_F_* flags. */
193 }
194 /** Sorted array of types, looked up using binary search! */
195 s_aTypes[] =
196 {
197#define STRMEM(str) sizeof(str) - 1, str
198 { STRMEM("Ci"), sizeof(RTINT), 10, RTSF_INT, RTSTR_F_VALSIGNED },
199 { STRMEM("Cp"), sizeof(RTGCPHYS), 16, RTSF_INTW, 0 },
200 { STRMEM("Cr"), sizeof(RTCCUINTREG), 16, RTSF_INTW, 0 },
201 { STRMEM("Cu"), sizeof(RTUINT), 10, RTSF_INT, 0 },
202 { STRMEM("Cv"), sizeof(void *), 16, RTSF_INTW, 0 },
203 { STRMEM("Cx"), sizeof(RTUINT), 16, RTSF_INT, 0 },
204 { STRMEM("Gi"), sizeof(RTGCINT), 10, RTSF_INT, RTSTR_F_VALSIGNED },
205 { STRMEM("Gp"), sizeof(RTGCPHYS), 16, RTSF_INTW, 0 },
206 { STRMEM("Gr"), sizeof(RTGCUINTREG), 16, RTSF_INTW, 0 },
207 { STRMEM("Gu"), sizeof(RTGCUINT), 10, RTSF_INT, 0 },
208 { STRMEM("Gv"), sizeof(RTGCPTR), 16, RTSF_INTW, 0 },
209 { STRMEM("Gx"), sizeof(RTGCUINT), 16, RTSF_INT, 0 },
210 { STRMEM("Hi"), sizeof(RTHCINT), 10, RTSF_INT, RTSTR_F_VALSIGNED },
211 { STRMEM("Hp"), sizeof(RTHCPHYS), 16, RTSF_INTW, 0 },
212 { STRMEM("Hr"), sizeof(RTGCUINTREG), 16, RTSF_INTW, 0 },
213 { STRMEM("Hu"), sizeof(RTHCUINT), 10, RTSF_INT, 0 },
214 { STRMEM("Hv"), sizeof(RTHCPTR), 16, RTSF_INTW, 0 },
215 { STRMEM("Hx"), sizeof(RTHCUINT), 16, RTSF_INT, 0 },
216 { STRMEM("I16"), sizeof(int16_t), 10, RTSF_INT, RTSTR_F_VALSIGNED },
217 { STRMEM("I32"), sizeof(int32_t), 10, RTSF_INT, RTSTR_F_VALSIGNED },
218 { STRMEM("I64"), sizeof(int64_t), 10, RTSF_INT, RTSTR_F_VALSIGNED },
219 { STRMEM("I8"), sizeof(int8_t), 10, RTSF_INT, RTSTR_F_VALSIGNED },
220 { STRMEM("Tbool"), sizeof(bool), 10, RTSF_BOOL, 0 },
221 { STRMEM("Tfile"), sizeof(RTFILE), 10, RTSF_INT, 0 },
222 { STRMEM("Tfmode"), sizeof(RTFMODE), 16, RTSF_INTW, 0 },
223 { STRMEM("Tfoff"), sizeof(RTFOFF), 10, RTSF_INT, RTSTR_F_VALSIGNED },
224 { STRMEM("Tfp16"), sizeof(RTFAR16), 16, RTSF_FP16, RTSTR_F_ZEROPAD },
225 { STRMEM("Tfp32"), sizeof(RTFAR32), 16, RTSF_FP32, RTSTR_F_ZEROPAD },
226 { STRMEM("Tfp64"), sizeof(RTFAR64), 16, RTSF_FP64, RTSTR_F_ZEROPAD },
227 { STRMEM("Tgid"), sizeof(RTGID), 10, RTSF_INT, RTSTR_F_VALSIGNED },
228 { STRMEM("Tino"), sizeof(RTINODE), 16, RTSF_INTW, 0 },
229 { STRMEM("Tint"), sizeof(RTINT), 10, RTSF_INT, RTSTR_F_VALSIGNED },
230 { STRMEM("Tiop"), sizeof(RTIOPORT), 16, RTSF_INTW, 0 },
231 { STRMEM("Tldrm"), sizeof(RTLDRMOD), 16, RTSF_INTW, 0 },
232 { STRMEM("Tnthrd"), sizeof(RTNATIVETHREAD), 16, RTSF_INTW, 0 },
233 { STRMEM("Tproc"), sizeof(RTPROCESS), 16, RTSF_INTW, 0 },
234 { STRMEM("Tptr"), sizeof(RTUINTPTR), 16, RTSF_INTW, 0 },
235 { STRMEM("Treg"), sizeof(RTUINTREG), 16, RTSF_INTW, 0 },
236 { STRMEM("Tsel"), sizeof(RTSEL), 16, RTSF_INTW, 0 },
237 { STRMEM("Tsem"), sizeof(RTSEMEVENT), 16, RTSF_INTW, 0 },
238 { STRMEM("Tsock"), sizeof(RTSOCKET), 10, RTSF_INT, 0 },
239 { STRMEM("Tthrd"), sizeof(RTTHREAD), 16, RTSF_INTW, 0 },
240 { STRMEM("Tuid"), sizeof(RTUID), 10, RTSF_INT, RTSTR_F_VALSIGNED },
241 { STRMEM("Tuint"), sizeof(RTUINT), 10, RTSF_INT, 0 },
242 { STRMEM("Tunicp"), sizeof(RTUNICP), 16, RTSF_INTW, RTSTR_F_ZEROPAD },
243 { STRMEM("Tutf16"), sizeof(RTUTF16), 16, RTSF_INTW, RTSTR_F_ZEROPAD },
244 { STRMEM("Tuuid"), sizeof(PCRTUUID), 16, RTSF_UUID, 0 },
245 { STRMEM("Txint"), sizeof(RTUINT), 16, RTSF_INT, 0 },
246 { STRMEM("U16"), sizeof(uint16_t), 10, RTSF_INT, 0 },
247 { STRMEM("U32"), sizeof(uint32_t), 10, RTSF_INT, 0 },
248 { STRMEM("U64"), sizeof(uint64_t), 10, RTSF_INT, 0 },
249 { STRMEM("U8"), sizeof(uint8_t), 10, RTSF_INT, 0 },
250 { STRMEM("X16"), sizeof(uint16_t), 16, RTSF_INT, 0 },
251 { STRMEM("X32"), sizeof(uint32_t), 16, RTSF_INT, 0 },
252 { STRMEM("X64"), sizeof(uint64_t), 16, RTSF_INT, 0 },
253 { STRMEM("X8"), sizeof(uint8_t), 16, RTSF_INT, 0 },
254#undef STRMEM
255 };
256 const char *pszType = *ppszFormat - 1;
257 int iStart = 0;
258 int iEnd = ELEMENTS(s_aTypes) - 1;
259 int i = ELEMENTS(s_aTypes) / 2;
260
261 union
262 {
263 uint8_t u8;
264 uint16_t u16;
265 uint32_t u32;
266 uint64_t u64;
267 int8_t i8;
268 int16_t i16;
269 int32_t i32;
270 int64_t i64;
271 RTFAR16 fp16;
272 RTFAR32 fp32;
273 RTFAR64 fp64;
274 bool fBool;
275 PCRTUUID pUuid;
276 } u;
277 char szBuf[80];
278 unsigned cch;
279
280 AssertMsg(!chArgSize, ("Not argument size '%c' for RT types! '%.10s'\n", chArgSize, pszFormatOrg));
281
282 /*
283 * Lookup the type - binary search.
284 */
285 for (;;)
286 {
287 int iDiff = strncmp(pszType, s_aTypes[i].sz, s_aTypes[i].cch);
288 if (!iDiff)
289 break;
290 if (iEnd == iStart)
291 {
292 AssertMsgFailed(("Invalid format type '%.10s'!\n", pszFormatOrg));
293 return 0;
294 }
295 if (iDiff < 0)
296 iEnd = i - 1;
297 else
298 iStart = i + 1;
299 if (iEnd < iStart)
300 {
301 AssertMsgFailed(("Invalid format type '%.10s'!\n", pszFormatOrg));
302 return 0;
303 }
304 i = iStart + (iEnd - iStart) / 2;
305 }
306
307 /*
308 * Advance the format string and merge flags.
309 */
310 *ppszFormat += s_aTypes[i].cch - 1;
311 fFlags |= s_aTypes[i].fFlags;
312
313 /*
314 * Fetch the argument.
315 * It's important that a signed value gets sign-extended up to 64-bit.
316 */
317 u.u64 = 0;
318 if (fFlags & RTSTR_F_VALSIGNED)
319 {
320 switch (s_aTypes[i].cb)
321 {
322 case sizeof(int8_t):
323 u.i64 = va_arg(*pArgs, /*int8_t*/int);
324 fFlags |= RTSTR_F_8BIT;
325 break;
326 case sizeof(int16_t):
327 u.i64 = va_arg(*pArgs, /*int16_t*/int);
328 fFlags |= RTSTR_F_16BIT;
329 break;
330 case sizeof(int32_t):
331 u.i64 = va_arg(*pArgs, int32_t);
332 fFlags |= RTSTR_F_32BIT;
333 break;
334 case sizeof(int64_t):
335 u.i64 = va_arg(*pArgs, int64_t);
336 fFlags |= RTSTR_F_64BIT;
337 break;
338 default:
339 AssertMsgFailed(("Invalid format error, size %d'!\n", s_aTypes[i].cb));
340 break;
341 }
342 }
343 else
344 {
345 switch (s_aTypes[i].cb)
346 {
347 case sizeof(uint8_t):
348 u.u8 = va_arg(*pArgs, /*uint8_t*/unsigned);
349 fFlags |= RTSTR_F_8BIT;
350 break;
351 case sizeof(uint16_t):
352 u.u16 = va_arg(*pArgs, /*uint16_t*/unsigned);
353 fFlags |= RTSTR_F_16BIT;
354 break;
355 case sizeof(uint32_t):
356 u.u32 = va_arg(*pArgs, uint32_t);
357 fFlags |= RTSTR_F_32BIT;
358 break;
359 case sizeof(uint64_t):
360 u.u64 = va_arg(*pArgs, uint64_t);
361 fFlags |= RTSTR_F_64BIT;
362 break;
363 case sizeof(RTFAR32):
364 u.fp32 = va_arg(*pArgs, RTFAR32);
365 break;
366 case sizeof(RTFAR64):
367 u.fp64 = va_arg(*pArgs, RTFAR64);
368 break;
369 default:
370 AssertMsgFailed(("Invalid format error, size %d'!\n", s_aTypes[i].cb));
371 break;
372 }
373 }
374
375 /*
376 * Format the output.
377 */
378 switch (s_aTypes[i].enmFormat)
379 {
380 case RTSF_INT:
381 {
382 cch = RTStrFormatNumber(szBuf, u.u64, s_aTypes[i].u8Base, cchWidth, cchPrecision, fFlags);
383 break;
384 }
385
386 /* hex which defaults to max width. */
387 case RTSF_INTW:
388 {
389 Assert(s_aTypes[i].u8Base == 16);
390 if (cchWidth < 0)
391 {
392 cchWidth = s_aTypes[i].cb * 2 + (fFlags & RTSTR_F_SPECIAL ? 2 : 0);
393 fFlags |= RTSTR_F_ZEROPAD;
394 }
395 cch = RTStrFormatNumber(szBuf, u.u64, s_aTypes[i].u8Base, cchWidth, cchPrecision, fFlags);
396 break;
397 }
398
399 case RTSF_FP16:
400 {
401 fFlags &= ~(RTSTR_F_VALSIGNED | RTSTR_F_BIT_MASK | RTSTR_F_WIDTH | RTSTR_F_PRECISION);
402 cch = RTStrFormatNumber(&szBuf[0], u.fp16.sel, 16, 4, -1, fFlags | RTSTR_F_16BIT);
403 Assert(cch == 4);
404 szBuf[4] = ':';
405 cch = RTStrFormatNumber(&szBuf[5], u.fp16.off, 16, 4, -1, fFlags | RTSTR_F_16BIT);
406 Assert(cch == 4);
407 cch = 4 + 1 + 4;
408 break;
409 }
410 case RTSF_FP32:
411 {
412 fFlags &= ~(RTSTR_F_VALSIGNED | RTSTR_F_BIT_MASK | RTSTR_F_WIDTH | RTSTR_F_PRECISION);
413 cch = RTStrFormatNumber(&szBuf[0], u.fp32.sel, 16, 4, -1, fFlags | RTSTR_F_16BIT);
414 Assert(cch == 4);
415 szBuf[4] = ':';
416 cch = RTStrFormatNumber(&szBuf[5], u.fp32.off, 16, 8, -1, fFlags | RTSTR_F_32BIT);
417 Assert(cch == 8);
418 cch = 4 + 1 + 8;
419 break;
420 }
421 case RTSF_FP64:
422 {
423 fFlags &= ~(RTSTR_F_VALSIGNED | RTSTR_F_BIT_MASK | RTSTR_F_WIDTH | RTSTR_F_PRECISION);
424 cch = RTStrFormatNumber(&szBuf[0], u.fp64.sel, 16, 4, -1, fFlags | RTSTR_F_16BIT);
425 Assert(cch == 4);
426 szBuf[4] = ':';
427 cch = RTStrFormatNumber(&szBuf[5], u.fp64.off, 16, 16, -1, fFlags | RTSTR_F_64BIT);
428 Assert(cch == 16);
429 cch = 4 + 1 + 16;
430 break;
431 }
432
433 case RTSF_UUID:
434 {
435 static const char szNull[] = "<NULL>";
436
437 if (VALID_PTR(u.pUuid))
438 {
439 /* cannot call RTUuidToStr because of GC/R0. */
440 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0,
441 "%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x",
442 u.pUuid->Gen.u32TimeLow,
443 u.pUuid->Gen.u16TimeMid,
444 u.pUuid->Gen.u16TimeHiAndVersion,
445 u.pUuid->Gen.u16ClockSeq & 0xff,
446 u.pUuid->Gen.u16ClockSeq >> 8,
447 u.pUuid->Gen.au8Node[0],
448 u.pUuid->Gen.au8Node[1],
449 u.pUuid->Gen.au8Node[2],
450 u.pUuid->Gen.au8Node[3],
451 u.pUuid->Gen.au8Node[4],
452 u.pUuid->Gen.au8Node[5]);
453 }
454 return pfnOutput(pvArgOutput, szNull, sizeof(szNull) - 1);
455 }
456
457 case RTSF_BOOL:
458 {
459 static const char szTrue[] = "true ";
460 static const char szFalse[] = "false";
461 if (u.u64 == 1)
462 return pfnOutput(pvArgOutput, szTrue, sizeof(szTrue) - 1);
463 if (u.u64 == 0)
464 return pfnOutput(pvArgOutput, szFalse, sizeof(szFalse) - 1);
465 /* invalid boolean value */
466 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "!%lld!", u.u64);
467 }
468
469 default:
470 AssertMsgFailed(("Internal error %d\n", s_aTypes[i].enmFormat));
471 return 0;
472 }
473
474 /*
475 * Finally, output the formatted string and return.
476 */
477 return pfnOutput(pvArgOutput, szBuf, cch);
478 }
479
480
481 /* Group 3 */
482
483 /*
484 * hex dumping.
485 */
486 case 'h':
487 {
488 char ch = *(*ppszFormat)++;
489 switch (ch)
490 {
491 /*
492 * Hex stuff.
493 */
494 case 'x':
495 {
496 uint8_t *pu8 = va_arg(*pArgs, uint8_t *);
497 if (cchWidth <= 0)
498 cchWidth = 16;
499 if (pu8)
500 {
501 ch = *(*ppszFormat)++;
502 switch (ch)
503 {
504 /*
505 * Regular hex dump.
506 */
507 case 'd':
508 {
509 size_t cch = 0;
510 int off = 0;
511
512 if (cchPrecision <= 0)
513 cchPrecision = 16;
514
515 while (off < cchWidth)
516 {
517 int i;
518 cch += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%s%0*x %04x:", off ? "\n" : "", sizeof(pu8) * 2, (uintptr_t)pu8, off);
519 for (i = 0; i < cchPrecision && off + i < cchWidth ; i++)
520 cch += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0,
521 off + i < cchWidth ? !(i & 7) && i ? "-%02x" : " %02x" : " ", pu8[i]);
522 while (i++ < cchPrecision)
523 cch += pfnOutput(pvArgOutput, " ", 3);
524
525 cch += pfnOutput(pvArgOutput, " ", 1);
526
527 for (i = 0; i < cchPrecision && off + i < cchWidth; i++)
528 {
529 uint8_t u8 = pu8[i];
530 cch += pfnOutput(pvArgOutput, u8 < 127 && u8 >= 32 ? (const char *)&u8 : ".", 1);
531 }
532
533 /* next */
534 pu8 += cchPrecision;
535 off += cchPrecision;
536 }
537 return cch;
538 }
539
540 /*
541 * Hex string.
542 */
543 case 's':
544 {
545 if (cchWidth-- > 0)
546 {
547 size_t cch = RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%02x", *pu8++);
548 for (; cchWidth > 0; cchWidth--, pu8++)
549 cch += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, " %02x", *pu8);
550 return cch;
551 }
552 break;
553 }
554
555 default:
556 AssertMsgFailed(("Invalid status code format type '%.10s'!\n", ch, pszFormatOrg));
557 break;
558 }
559 }
560 else
561 return pfnOutput(pvArgOutput, "<null>", sizeof("<null>") - 1);
562 break;
563 }
564
565 default:
566 AssertMsgFailed(("Invalid status code format type '%.10s'!\n", ch, pszFormatOrg));
567 return 0;
568
569 }
570 break;
571 }
572
573 /*
574 * iprt status code: %Vrc, %Vrs, %Vrf, %Vra.
575 */
576 case 'r':
577 {
578 int rc = va_arg(*pArgs, int);
579 char ch = *(*ppszFormat)++;
580#ifdef IN_RING3 /* we don't want this anywhere else yet. */
581 PCRTSTATUSMSG pMsg = RTErrGet(rc);
582 switch (ch)
583 {
584 case 'c':
585 return pfnOutput(pvArgOutput, pMsg->pszDefine, strlen(pMsg->pszDefine));
586 case 's':
587 return pfnOutput(pvArgOutput, pMsg->pszMsgShort, strlen(pMsg->pszMsgShort));
588 case 'f':
589 return pfnOutput(pvArgOutput, pMsg->pszMsgFull, strlen(pMsg->pszMsgFull));
590 case 'a':
591 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%s (%d) - %s", pMsg->pszDefine, rc, pMsg->pszMsgFull);
592 default:
593 AssertMsgFailed(("Invalid status code format type '%.10s'!\n", pszFormatOrg));
594 return 0;
595 }
596#else /* !IN_RING3 */
597 switch (ch)
598 {
599 case 'c':
600 case 's':
601 case 'f':
602 case 'a':
603 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%d", rc);
604 default:
605 AssertMsgFailed(("Invalid status code format type '%.10s'!\n", pszFormatOrg));
606 return 0;
607 }
608#endif /* !IN_RING3 */
609 break;
610 }
611
612#if defined(IN_RING3)
613 /*
614 * Windows status code: %Rwc, %Rwf, %Rwa
615 */
616 case 'w':
617 {
618 long rc = va_arg(*pArgs, long);
619 char ch = *(*ppszFormat)++;
620#if defined(RT_OS_WINDOWS)
621 PCRTWINERRMSG pMsg = RTErrWinGet(rc);
622#endif
623 switch (ch)
624 {
625#if defined(RT_OS_WINDOWS)
626 case 'c':
627 return pfnOutput(pvArgOutput, pMsg->pszDefine, strlen(pMsg->pszDefine));
628 case 'f':
629 return pfnOutput(pvArgOutput, pMsg->pszMsgFull,strlen(pMsg->pszMsgFull));
630 case 'a':
631 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "%s (0x%08X) - %s", pMsg->pszDefine, rc, pMsg->pszMsgFull);
632#else
633 case 'c':
634 case 'f':
635 case 'a':
636 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "0x%08X", rc);
637#endif
638 default:
639 AssertMsgFailed(("Invalid status code format type '%.10s'!\n", pszFormatOrg));
640 return 0;
641 }
642 break;
643 }
644#endif /* IN_RING3 */
645
646 /*
647 * Group 4, structure dumpers.
648 */
649 case 'D':
650 {
651 /*
652 * Interpret the type.
653 */
654 typedef enum
655 {
656 RTST_TIMESPEC
657 } RTST;
658/** Set if it's a pointer */
659#define RTST_FLAGS_POINTER RT_BIT(0)
660 static const struct
661 {
662 uint8_t cch; /**< the length of the string. */
663 char sz[16-2]; /**< the part following 'R'. */
664 uint8_t cb; /**< the size of the argument. */
665 uint8_t fFlags; /**< RTST_FLAGS_* */
666 RTST enmType; /**< The structure type. */
667 }
668 /** Sorted array of types, looked up using binary search! */
669 s_aTypes[] =
670 {
671#define STRMEM(str) sizeof(str) - 1, str
672 { STRMEM("Dtimespec"), sizeof(PCRTTIMESPEC), RTST_FLAGS_POINTER, RTST_TIMESPEC},
673#undef STRMEM
674 };
675 const char *pszType = *ppszFormat - 1;
676 int iStart = 0;
677 int iEnd = ELEMENTS(s_aTypes) - 1;
678 int i = ELEMENTS(s_aTypes) / 2;
679
680 union
681 {
682 const void *pv;
683 uint64_t u64;
684 PCRTTIMESPEC pTimeSpec;
685 } u;
686
687 AssertMsg(!chArgSize, ("Not argument size '%c' for RT types! '%.10s'\n", chArgSize, pszFormatOrg));
688
689 /*
690 * Lookup the type - binary search.
691 */
692 for (;;)
693 {
694 int iDiff = strncmp(pszType, s_aTypes[i].sz, s_aTypes[i].cch);
695 if (!iDiff)
696 break;
697 if (iEnd == iStart)
698 {
699 AssertMsgFailed(("Invalid format type '%.10s'!\n", pszFormatOrg));
700 return 0;
701 }
702 if (iDiff < 0)
703 iEnd = i - 1;
704 else
705 iStart = i + 1;
706 if (iEnd < iStart)
707 {
708 AssertMsgFailed(("Invalid format type '%.10s'!\n", pszFormatOrg));
709 return 0;
710 }
711 i = iStart + (iEnd - iStart) / 2;
712 }
713 *ppszFormat += s_aTypes[i].cch - 1;
714
715 /*
716 * Fetch the argument.
717 */
718 u.u64 = 0;
719 switch (s_aTypes[i].cb)
720 {
721 case sizeof(const void *):
722 u.pv = va_arg(*pArgs, const void *);
723 break;
724 default:
725 AssertMsgFailed(("Invalid format error, size %d'!\n", s_aTypes[i].cb));
726 break;
727 }
728
729 /*
730 * If it's a pointer, we'll check if it's valid before going on.
731 */
732 if ((s_aTypes[i].fFlags & RTST_FLAGS_POINTER) && !VALID_PTR(u.pv))
733 return pfnOutput(pvArgOutput, "<null>", sizeof("<null>") - 1);
734
735 /*
736 * Format the output.
737 */
738 switch (s_aTypes[i].enmType)
739 {
740 case RTST_TIMESPEC:
741 return RTStrFormat(pfnOutput, pvArgOutput, NULL, NULL, "%lld ns", RTTimeSpecGetNano(u.pTimeSpec));
742
743 default:
744 AssertMsgFailed(("Invalid/unhandled enmType=%d\n", s_aTypes[i].enmType));
745 break;
746 }
747 break;
748 }
749
750 /*
751 * Invalid/Unknown. Bitch about it.
752 */
753 default:
754 AssertMsgFailed(("Invalid VBox format type '%.10s'!\n", pszFormatOrg));
755 break;
756 }
757 }
758 else
759 AssertMsgFailed(("Invalid VBox format type '%.10s'!\n", pszFormatOrg));
760
761 NOREF(pszFormatOrg);
762 return 0;
763}
764
765
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