VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/string/strformat.cpp@ 9552

Last change on this file since 9552 was 8619, checked in by vboxsync, 17 years ago

New nesting extension '%M' that replaces the format string. Note that it won't currently process anything after the '%M' in the orignal string, but this is sufficient for joining log messages.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 32.1 KB
Line 
1/* $Id: strformat.cpp 8619 2008-05-06 11:16:36Z vboxsync $ */
2/** @file
3 * IPRT - String Formatter.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 *
26 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Defined Constants *
34*******************************************************************************/
35#define ISDIGIT(c) ((c) >= '0' && (c) <= '9')
36/*#define MAX(a, b) ((a) >= (b) ? (a) : (b))
37#define MIN(a, b) ((a) < (b) ? (a) : (b)) */
38
39
40/*******************************************************************************
41* Header Files *
42*******************************************************************************/
43#define LOG_GROUP RTLOGGROUP_STRING
44#include <iprt/string.h>
45#include <iprt/assert.h>
46#ifdef IN_RING3
47# include <iprt/alloc.h>
48# include <iprt/err.h>
49# include <iprt/uni.h>
50#endif
51#include <iprt/string.h>
52#include <iprt/stdarg.h>
53#include "internal/string.h"
54
55/* Wrappers for converting to iprt facilities. */
56#define SSToDS(ptr) ptr
57#define kASSERT Assert
58#define KENDIAN_LITTLE 1
59#define KENDIAN KENDIAN_LITTLE
60#define KSIZE size_t
61typedef struct
62{
63 uint32_t ulLo;
64 uint32_t ulHi;
65} KSIZE64;
66
67
68/*******************************************************************************
69* Internal Functions *
70*******************************************************************************/
71static unsigned _strnlen(const char *psz, unsigned cchMax);
72static unsigned _strnlenUtf16(PCRTUTF16 pwsz, unsigned cchMax);
73static int rtStrFormatNumber(char *psz, KSIZE64 ullValue, unsigned int uiBase, signed int cchWidth, signed int cchPrecision, unsigned int fFlags);
74
75
76/**
77 * Finds the length of a string up to cchMax.
78 * @returns Length.
79 * @param psz Pointer to string.
80 * @param cchMax Max length.
81 */
82static unsigned _strnlen(const char *psz, unsigned cchMax)
83{
84 const char *pszC = psz;
85
86 while (cchMax-- > 0 && *psz != '\0')
87 psz++;
88
89 return psz - pszC;
90}
91
92
93/**
94 * Finds the length of a string up to cchMax.
95 * @returns Length.
96 * @param pwsz Pointer to string.
97 * @param cchMax Max length.
98 */
99static unsigned _strnlenUtf16(PCRTUTF16 pwsz, unsigned cchMax)
100{
101#ifdef IN_RING3
102 unsigned cwc = 0;
103 while (cchMax-- > 0)
104 {
105 RTUNICP cp;
106 int rc = RTUtf16GetCpEx(&pwsz, &cp);
107 AssertRC(rc);
108 if (RT_FAILURE(rc) || !cp)
109 break;
110 cwc++;
111 }
112 return cwc;
113#else /* !IN_RING3 */
114 PCRTUTF16 pwszC = pwsz;
115
116 while (cchMax-- > 0 && *pwsz != '\0')
117 pwsz++;
118
119 return pwsz - pwszC;
120#endif /* !IN_RING3 */
121}
122
123
124/**
125 * Finds the length of a string up to cchMax.
126 * @returns Length.
127 * @param pusz Pointer to string.
128 * @param cchMax Max length.
129 */
130static unsigned _strnlenUni(PCRTUNICP pusz, unsigned cchMax)
131{
132 PCRTUNICP puszC = pusz;
133
134 while (cchMax-- > 0 && *pusz != '\0')
135 pusz++;
136
137 return pusz - puszC;
138}
139
140
141/**
142 * Formats an integer number according to the parameters.
143 *
144 * @returns Length of the formatted number.
145 * @param psz Pointer to output string buffer of sufficient size.
146 * @param u64Value Value to format.
147 * @param uiBase Number representation base.
148 * @param cchWidth Width.
149 * @param cchPrecision Precision.
150 * @param fFlags Flags (NTFS_*).
151 */
152RTDECL(int) RTStrFormatNumber(char *psz, uint64_t u64Value, unsigned int uiBase, signed int cchWidth, signed int cchPrecision, unsigned int fFlags)
153{
154 return rtStrFormatNumber(psz, *(KSIZE64 *)(void *)&u64Value, uiBase, cchWidth, cchPrecision, fFlags);
155}
156
157
158
159/**
160 * Formats an integer number according to the parameters.
161 *
162 * @returns Length of the number.
163 * @param psz Pointer to output string.
164 * @param ullValue Value. Using the high part is optional.
165 * @param uiBase Number representation base.
166 * @param cchWidth Width
167 * @param cchPrecision Precision.
168 * @param fFlags Flags (NTFS_*).
169 */
170static int rtStrFormatNumber(char *psz, KSIZE64 ullValue, unsigned int uiBase, signed int cchWidth, signed int cchPrecision, unsigned int fFlags)
171{
172 const char * pachDigits = "0123456789abcdef";
173 char * pszStart = psz;
174 int cchValue;
175 unsigned long ul;
176#if 0
177 unsigned long ullow;
178#endif
179 int i;
180 int j;
181
182/** @todo Formatting of 64 bit numbers is broken, fix it! */
183
184 /*
185 * Validate and addjust input...
186 */
187/** @todo r=bird: Dmitry, who is calling this code with uiBase == 0? */
188 if (uiBase == 0)
189 uiBase = 10;
190 kASSERT((uiBase >= 2 || uiBase <= 16));
191 if (fFlags & RTSTR_F_CAPITAL)
192 pachDigits = "0123456789ABCDEF";
193 if (fFlags & RTSTR_F_LEFT)
194 fFlags &= ~RTSTR_F_ZEROPAD;
195
196 /*
197 * Determin value length
198 */
199 cchValue = 0;
200 if (ullValue.ulHi || (fFlags & RTSTR_F_64BIT))
201 {
202 uint64_t u64 = *(uint64_t *)(void *)&ullValue;
203 if ((fFlags & RTSTR_F_VALSIGNED) && (ullValue.ulHi & 0x80000000))
204 u64 = -(int64_t)u64;
205 do
206 {
207 cchValue++;
208 u64 /= uiBase;
209 } while (u64);
210 }
211 else
212 {
213 ul = (fFlags & RTSTR_F_VALSIGNED) && (ullValue.ulLo & 0x80000000) ? -(int32_t)ullValue.ulLo : ullValue.ulLo;
214 do
215 {
216 cchValue++;
217 ul /= uiBase;
218 } while (ul);
219 }
220
221 /*
222 * Sign (+/-).
223 */
224 i = 0;
225 if (fFlags & RTSTR_F_VALSIGNED)
226 {
227 if ((ullValue.ulHi || (fFlags & RTSTR_F_64BIT) ? ullValue.ulHi : ullValue.ulLo) & 0x80000000)
228 {
229 ullValue.ulLo = -(int32_t)ullValue.ulLo;
230 if (ullValue.ulHi)
231 ullValue.ulHi = ~ullValue.ulHi;
232 psz[i++] = '-';
233 }
234 else if (fFlags & (RTSTR_F_PLUS | RTSTR_F_BLANK))
235 psz[i++] = (char)(fFlags & RTSTR_F_PLUS ? '+' : ' ');
236 }
237
238 /*
239 * Special (0/0x).
240 */
241 if ((fFlags & RTSTR_F_SPECIAL) && (uiBase % 8) == 0)
242 {
243 psz[i++] = '0';
244 if (uiBase == 16)
245 psz[i++] = (char)(fFlags & RTSTR_F_CAPITAL ? 'X' : 'x');
246 }
247
248 /*
249 * width - only if ZEROPAD
250 */
251 cchWidth -= i + cchValue;
252 if (fFlags & RTSTR_F_ZEROPAD)
253 while (--cchWidth >= 0)
254 {
255 psz[i++] = '0';
256 cchPrecision--;
257 }
258 else if (!(fFlags & RTSTR_F_LEFT) && cchWidth > 0)
259 {
260 for (j = i-1; j >= 0; j--)
261 psz[cchWidth + j] = psz[j];
262 for (j = 0; j < cchWidth; j++)
263 psz[j] = ' ';
264 i += cchWidth;
265 }
266 psz += i;
267
268
269 /*
270 * precision
271 */
272 while (--cchPrecision >= cchValue)
273 *psz++ = '0';
274
275 /*
276 * write number - not good enough but it works
277 */
278 psz += cchValue;
279 i = -1;
280 if (ullValue.ulHi || (fFlags & RTSTR_F_64BIT))
281 {
282 uint64_t u64 = *(uint64_t *)(void *)&ullValue;
283 do
284 {
285 psz[i--] = pachDigits[u64 % uiBase];
286 u64 /= uiBase;
287 } while (u64);
288 }
289 else
290 {
291 ul = (fFlags & RTSTR_F_VALSIGNED) && (ullValue.ulLo & 0x80000000) ? -(int32_t)ullValue.ulLo : ullValue.ulLo;
292 do
293 {
294 psz[i--] = pachDigits[ul % uiBase];
295 ul /= uiBase;
296 } while (ul);
297 }
298
299
300 /*
301 * width if RTSTR_F_LEFT
302 */
303 if (fFlags & RTSTR_F_LEFT)
304 while (--cchWidth >= 0)
305 *psz++ = ' ';
306
307 *psz = '\0';
308 return psz - pszStart;
309}
310
311
312/**
313 * Partial implementation of a printf like formatter.
314 * It doesn't do everything correct, and there is no floating point support.
315 * However, it supports custom formats by the means of a format callback.
316 *
317 * @returns number of bytes formatted.
318 * @param pfnOutput Output worker.
319 * Called in two ways. Normally with a string an it's length.
320 * For termination, it's called with NULL for string, 0 for length.
321 * @param pvArgOutput Argument to the output worker.
322 * @param pfnFormat Custom format worker.
323 * @param pvArgFormat Argument to the format worker.
324 * @param pszFormat Format string.
325 * @param InArgs Argument list.
326 */
327RTDECL(size_t) RTStrFormatV(PFNRTSTROUTPUT pfnOutput, void *pvArgOutput, PFNSTRFORMAT pfnFormat, void *pvArgFormat, const char *pszFormat, va_list InArgs)
328{
329 va_list args;
330 KSIZE cch = 0;
331 const char *pszStartOutput = pszFormat;
332
333 va_copy(args, InArgs); /* make a copy so we can reference it (AMD64 / gcc). */
334
335 while (*pszFormat != '\0')
336 {
337 if (*pszFormat == '%')
338 {
339 /* output pending string. */
340 if (pszStartOutput != pszFormat)
341 cch += pfnOutput(pvArgOutput, pszStartOutput, pszFormat - pszStartOutput);
342
343 /* skip '%' */
344 pszFormat++;
345 if (*pszFormat == '%') /* '%%'-> '%' */
346 pszStartOutput = pszFormat++;
347 else
348 {
349 unsigned int fFlags = 0;
350 int cchWidth = -1;
351 int cchPrecision = -1;
352 unsigned int uBase = 10;
353 char chArgSize;
354
355 /* flags */
356 for (;;)
357 {
358 switch (*pszFormat++)
359 {
360 case '#': fFlags |= RTSTR_F_SPECIAL; continue;
361 case '-': fFlags |= RTSTR_F_LEFT; continue;
362 case '+': fFlags |= RTSTR_F_PLUS; continue;
363 case ' ': fFlags |= RTSTR_F_BLANK; continue;
364 case '0': fFlags |= RTSTR_F_ZEROPAD; continue;
365 }
366 pszFormat--;
367 break;
368 }
369
370 /* width */
371 if (ISDIGIT(*pszFormat))
372 {
373 for (cchWidth = 0; ISDIGIT(*pszFormat); pszFormat++)
374 {
375 cchWidth *= 10;
376 cchWidth += *pszFormat - '0';
377 }
378 fFlags |= RTSTR_F_WIDTH;
379 }
380 else if (*pszFormat == '*')
381 {
382 pszFormat++;
383 cchWidth = va_arg(args, int);
384 if (cchWidth < 0)
385 {
386 cchWidth = -cchWidth;
387 fFlags |= RTSTR_F_LEFT;
388 }
389 fFlags |= RTSTR_F_WIDTH;
390 }
391
392 /* precision */
393 if (*pszFormat == '.')
394 {
395 pszFormat++;
396 if (ISDIGIT(*pszFormat))
397 {
398 for (cchPrecision = 0; ISDIGIT(*pszFormat); pszFormat++)
399 {
400 cchPrecision *= 10;
401 cchPrecision += *pszFormat - '0';
402 }
403
404 }
405 else if (*pszFormat == '*')
406 {
407 pszFormat++;
408 cchPrecision = va_arg(args, int);
409 }
410 if (cchPrecision < 0)
411 cchPrecision = 0;
412 fFlags |= RTSTR_F_PRECISION;
413 }
414
415 /* argsize */
416 chArgSize = *pszFormat;
417 if (chArgSize != 'l' && chArgSize != 'L' && chArgSize != 'h' && chArgSize != 'j' && chArgSize != 'z' && chArgSize != 't')
418 chArgSize = 0;
419 else
420 {
421 pszFormat++;
422 if (*pszFormat == 'l' && chArgSize == 'l')
423 {
424 chArgSize = 'L';
425 pszFormat++;
426 }
427 else if (*pszFormat == 'h' && chArgSize == 'h')
428 {
429 chArgSize = 'H';
430 pszFormat++;
431 }
432 }
433
434 /*
435 * The type.
436 */
437 switch (*pszFormat++)
438 {
439 /* char */
440 case 'c':
441 {
442 char ch;
443
444 if (!(fFlags & RTSTR_F_LEFT))
445 while (--cchWidth > 0)
446 cch += pfnOutput(pvArgOutput, " ", 1);
447
448 ch = (char)va_arg(args, int);
449 cch += pfnOutput(pvArgOutput, SSToDS(&ch), 1);
450
451 while (--cchWidth > 0)
452 cch += pfnOutput(pvArgOutput, " ", 1);
453 break;
454 }
455
456#ifndef IN_RING3
457 case 'S': /* Unicode string as current code page -> Unicode as UTF-8 in GC/R0. */
458 chArgSize = 'l';
459 /* fall thru */
460#endif
461 case 's': /* Unicode string as utf8 */
462 {
463 if (chArgSize == 'l')
464 {
465 /* utf-16 -> utf-8 */
466 int cchStr;
467 PCRTUTF16 pwszStr = va_arg(args, PRTUTF16);
468
469 if (!VALID_PTR(pwszStr))
470 {
471 static RTUTF16 s_wszNull[] = {'<', 'N', 'U', 'L', 'L', '>', '\0' };
472 pwszStr = s_wszNull;
473 }
474 cchStr = _strnlenUtf16(pwszStr, (unsigned)cchPrecision);
475 if (!(fFlags & RTSTR_F_LEFT))
476 while (--cchWidth >= cchStr)
477 cch += pfnOutput(pvArgOutput, " ", 1);
478 while (cchStr-- > 0)
479 {
480#ifdef IN_RING3
481 RTUNICP Cp;
482 RTUtf16GetCpEx(&pwszStr, &Cp);
483 char szUtf8[8]; /* Cp=0x7fffffff -> 6 bytes. */
484 char *pszEnd = RTStrPutCp(szUtf8, Cp);
485 cch += pfnOutput(pvArgOutput, szUtf8, pszEnd - szUtf8);
486#else
487 char ch = (char)*pwszStr++;
488 cch += pfnOutput(pvArgOutput, &ch, 1);
489#endif
490 }
491 while (--cchWidth >= cchStr)
492 cch += pfnOutput(pvArgOutput, " ", 1);
493 }
494 else if (chArgSize == 'L')
495 {
496 /* unicp -> utf8 */
497 int cchStr;
498 PCRTUNICP puszStr = va_arg(args, PCRTUNICP);
499
500 if (!VALID_PTR(puszStr))
501 {
502 static RTUNICP s_uszNull[] = {'<', 'N', 'U', 'L', 'L', '>', '\0' };
503 puszStr = s_uszNull;
504 }
505 cchStr = _strnlenUni(puszStr, (unsigned)cchPrecision);
506 if (!(fFlags & RTSTR_F_LEFT))
507 while (--cchWidth >= cchStr)
508 cch += pfnOutput(pvArgOutput, " ", 1);
509
510 while (cchStr-- > 0)
511 {
512#ifdef IN_RING3
513 char szUtf8[8]; /* Cp=0x7fffffff -> 6 bytes. */
514 char *pszEnd = RTStrPutCp(szUtf8, *puszStr++);
515 cch += pfnOutput(pvArgOutput, szUtf8, pszEnd - szUtf8);
516#else
517 char ch = (char)*puszStr++;
518 cch += pfnOutput(pvArgOutput, &ch, 1);
519#endif
520 }
521 while (--cchWidth >= cchStr)
522 cch += pfnOutput(pvArgOutput, " ", 1);
523 }
524 else
525 {
526 int cchStr;
527 const char *pszStr = va_arg(args, char*);
528
529 if (!VALID_PTR(pszStr))
530 pszStr = "<NULL>";
531 cchStr = _strnlen(pszStr, (unsigned)cchPrecision);
532 if (!(fFlags & RTSTR_F_LEFT))
533 while (--cchWidth >= cchStr)
534 cch += pfnOutput(pvArgOutput, " ", 1);
535
536 cch += pfnOutput(pvArgOutput, pszStr, cchStr);
537
538 while (--cchWidth >= cchStr)
539 cch += pfnOutput(pvArgOutput, " ", 1);
540 }
541 break;
542 }
543
544#ifdef IN_RING3
545 case 'S': /* Unicode string as current code page. */
546 {
547 if (chArgSize == 'l')
548 {
549 /* UTF-16 */
550 int cchStr;
551 PCRTUTF16 pwsz2Str = va_arg(args, PRTUTF16);
552 if (!VALID_PTR(pwsz2Str))
553 {
554 static RTUTF16 s_wsz2Null[] = {'<', 'N', 'U', 'L', 'L', '>', '\0' };
555 pwsz2Str = s_wsz2Null;
556 }
557
558 cchStr = _strnlenUtf16(pwsz2Str, (unsigned)cchPrecision);
559 if (!(fFlags & RTSTR_F_LEFT))
560 while (--cchWidth >= cchStr)
561 cch += pfnOutput(pvArgOutput, " ", 1);
562
563 if (cchStr)
564 {
565 /* allocate temporary buffer. */
566 PRTUTF16 pwsz2Tmp = (PRTUTF16)RTMemTmpAlloc((cchStr + 1) * sizeof(RTUTF16));
567 memcpy(pwsz2Tmp, pwsz2Str, cchStr * sizeof(RTUTF16));
568 pwsz2Tmp[cchStr] = '\0';
569
570 char *pszUtf8;
571 int rc = RTUtf16ToUtf8(pwsz2Tmp, &pszUtf8);
572 if (RT_SUCCESS(rc))
573 {
574 char *pszCurCp;
575 rc = RTStrUtf8ToCurrentCP(&pszCurCp, pszUtf8);
576 if (RT_SUCCESS(rc))
577 {
578 cch += pfnOutput(pvArgOutput, pszCurCp, strlen(pszCurCp));
579 RTStrFree(pszCurCp);
580 }
581 RTStrFree(pszUtf8);
582 }
583 if (RT_FAILURE(rc))
584 while (cchStr-- > 0)
585 cch += pfnOutput(pvArgOutput, "\x7f", 1);
586 RTMemTmpFree(pwsz2Tmp);
587 }
588
589 while (--cchWidth >= cchStr)
590 cch += pfnOutput(pvArgOutput, " ", 1);
591 }
592 else if (chArgSize == 'L')
593 {
594 /* UCS-32 */
595 AssertMsgFailed(("Not implemented yet\n"));
596 }
597 else
598 {
599 /* UTF-8 */
600 int cchStr;
601 const char *pszStr = va_arg(args, char *);
602
603 if (!VALID_PTR(pszStr))
604 pszStr = "<NULL>";
605 cchStr = _strnlen(pszStr, (unsigned)cchPrecision);
606 if (!(fFlags & RTSTR_F_LEFT))
607 while (--cchWidth >= cchStr)
608 cch += pfnOutput(pvArgOutput, " ", 1);
609
610 if (cchStr)
611 {
612 /* allocate temporary buffer. */
613 char *pszTmp = (char *)RTMemTmpAlloc(cchStr + 1);
614 memcpy(pszTmp, pszStr, cchStr);
615 pszTmp[cchStr] = '\0';
616
617 char *pszCurCp;
618 int rc = RTStrUtf8ToCurrentCP(&pszCurCp, pszTmp);
619 if (RT_SUCCESS(rc))
620 {
621 cch += pfnOutput(pvArgOutput, pszCurCp, strlen(pszCurCp));
622 RTStrFree(pszCurCp);
623 }
624 else
625 while (cchStr-- > 0)
626 cch += pfnOutput(pvArgOutput, "\x7f", 1);
627 RTMemTmpFree(pszTmp);
628 }
629
630 while (--cchWidth >= cchStr)
631 cch += pfnOutput(pvArgOutput, " ", 1);
632 }
633 break;
634 }
635#endif
636
637
638 /*-----------------*/
639 /* integer/pointer */
640 /*-----------------*/
641 case 'd':
642 case 'i':
643 case 'o':
644 case 'p':
645 case 'u':
646 case 'x':
647 case 'X':
648 {
649 char achNum[64]; /* FIXME */
650 int cchNum;
651 uint64_t u64Value;
652
653 switch (pszFormat[-1])
654 {
655 case 'd': /* signed decimal integer */
656 case 'i':
657 fFlags |= RTSTR_F_VALSIGNED;
658 break;
659
660 case 'o':
661 uBase = 8;
662 break;
663
664 case 'p':
665 fFlags |= RTSTR_F_ZEROPAD; /* Note not standard behaviour (but I like it this way!) */
666 uBase = 16;
667 if (cchWidth < 0)
668 cchWidth = sizeof(char *) * 2;
669 break;
670
671 case 'u':
672 uBase = 10;
673 break;
674
675 case 'X':
676 fFlags |= RTSTR_F_CAPITAL;
677 case 'x':
678 uBase = 16;
679 break;
680 }
681
682 if (pszFormat[-1] == 'p')
683 u64Value = va_arg(args, uintptr_t);
684 else if (fFlags & RTSTR_F_VALSIGNED)
685 {
686 if (chArgSize == 'L')
687 {
688 u64Value = va_arg(args, int64_t);
689 fFlags |= RTSTR_F_64BIT;
690 }
691 else if (chArgSize == 'l')
692 {
693 u64Value = va_arg(args, signed long);
694 fFlags |= RTSTR_GET_BIT_FLAG(unsigned long);
695 }
696 else if (chArgSize == 'h')
697 {
698 u64Value = va_arg(args, /* signed short */ int);
699 fFlags |= RTSTR_GET_BIT_FLAG(signed short);
700 }
701 else if (chArgSize == 'H')
702 {
703 u64Value = va_arg(args, /* int8_t */ int);
704 fFlags |= RTSTR_GET_BIT_FLAG(int8_t);
705 }
706 else if (chArgSize == 'j')
707 {
708 u64Value = va_arg(args, /*intmax_t*/ int64_t);
709 fFlags |= RTSTR_F_64BIT;
710 }
711 else if (chArgSize == 'z')
712 {
713 u64Value = va_arg(args, size_t);
714 fFlags |= RTSTR_GET_BIT_FLAG(size_t);
715 }
716 else if (chArgSize == 't')
717 {
718 u64Value = va_arg(args, ptrdiff_t);
719 fFlags |= RTSTR_GET_BIT_FLAG(ptrdiff_t);
720 }
721 else
722 {
723 u64Value = va_arg(args, signed int);
724 fFlags |= RTSTR_GET_BIT_FLAG(signed int);
725 }
726 }
727 else
728 {
729 if (chArgSize == 'L')
730 {
731 u64Value = va_arg(args, uint64_t);
732 fFlags |= RTSTR_F_64BIT;
733 }
734 else if (chArgSize == 'l')
735 {
736 u64Value = va_arg(args, unsigned long);
737 fFlags |= RTSTR_GET_BIT_FLAG(unsigned long);
738 }
739 else if (chArgSize == 'h')
740 {
741 u64Value = va_arg(args, /* unsigned short */ int);
742 fFlags |= RTSTR_GET_BIT_FLAG(unsigned short);
743 }
744 else if (chArgSize == 'H')
745 {
746 u64Value = va_arg(args, /* uint8_t */ int);
747 fFlags |= RTSTR_GET_BIT_FLAG(uint8_t);
748 }
749 else if (chArgSize == 'j')
750 {
751 u64Value = va_arg(args, /*uintmax_t*/ int64_t);
752 fFlags |= RTSTR_F_64BIT;
753 }
754 else if (chArgSize == 'z')
755 {
756 u64Value = va_arg(args, size_t);
757 fFlags |= RTSTR_GET_BIT_FLAG(size_t);
758 }
759 else if (chArgSize == 't')
760 {
761 u64Value = va_arg(args, ptrdiff_t);
762 fFlags |= RTSTR_GET_BIT_FLAG(ptrdiff_t);
763 }
764 else
765 {
766 u64Value = va_arg(args, unsigned int);
767 fFlags |= RTSTR_GET_BIT_FLAG(unsigned int);
768 }
769 }
770 cchNum = RTStrFormatNumber((char *)SSToDS(&achNum), u64Value, uBase, cchWidth, cchPrecision, fFlags);
771 cch += pfnOutput(pvArgOutput, (char *)SSToDS(&achNum), cchNum);
772 break;
773 }
774
775 /*
776 * Nested extensions.
777 */
778 case 'M': /* replace the format string (not stacked yet). */
779 {
780 pszStartOutput = pszFormat = va_arg(args, const char *);
781 AssertPtr(pszStartOutput);
782 break;
783 }
784
785 case 'N': /* real nesting. */
786 {
787 const char *pszFormatNested = va_arg(args, const char *);
788 va_list *pArgsNested = va_arg(args, va_list *);
789 va_list ArgsNested;
790 va_copy(ArgsNested, *pArgsNested);
791 Assert(pszFormatNested);
792 cch += RTStrFormatV(pfnOutput, pvArgOutput, pfnFormat, pvArgFormat, pszFormatNested, ArgsNested);
793 break;
794 }
795
796 /*
797 * IPRT Extensions.
798 */
799 case 'R':
800 {
801 if (*pszFormat != '[')
802 {
803 pszFormat--;
804 cch += rtstrFormatRt(pfnOutput, pvArgOutput, &pszFormat, &args, cchPrecision, cchWidth, fFlags, chArgSize);
805 }
806 else
807 {
808 pszFormat--;
809 cch += rtstrFormatType(pfnOutput, pvArgOutput, &pszFormat, &args, cchPrecision, cchWidth, fFlags, chArgSize);
810 }
811 break;
812 }
813
814#ifdef RT_WITH_VBOX
815 /*
816 * VBox extensions.
817 */
818 case 'V':
819 {
820 pszFormat--;
821 cch += rtstrFormatVBox(pfnOutput, pvArgOutput, &pszFormat, &args, cchPrecision, cchWidth, fFlags, chArgSize);
822 break;
823 }
824#endif
825
826 /*
827 * Custom format.
828 */
829 default:
830 {
831 if (pfnFormat)
832 {
833 pszFormat--;
834 cch += pfnFormat(pvArgFormat, pfnOutput, pvArgOutput, &pszFormat, &args, cchPrecision, cchWidth, fFlags, chArgSize);
835 }
836 break;
837 }
838 }
839 pszStartOutput = pszFormat;
840 }
841 }
842 else
843 pszFormat++;
844 }
845
846 /* output pending string. */
847 if (pszStartOutput != pszFormat)
848 cch += pfnOutput(pvArgOutput, pszStartOutput, pszFormat - pszStartOutput);
849
850 /* terminate the output */
851 pfnOutput(pvArgOutput, NULL, 0);
852
853 return cch;
854}
855
856
857/**
858 * Partial implementation of a printf like formatter.
859 * It doesn't do everything correct, and there is no floating point support.
860 * However, it supports custom formats by the means of a format callback.
861 *
862 * @returns number of bytes formatted.
863 * @param pfnOutput Output worker.
864 * Called in two ways. Normally with a string an it's length.
865 * For termination, it's called with NULL for string, 0 for length.
866 * @param pvArgOutput Argument to the output worker.
867 * @param pfnFormat Custom format worker.
868 * @param pvArgFormat Argument to the format worker.
869 * @param pszFormat Format string.
870 * @param ... Argument list.
871 */
872RTDECL(size_t) RTStrFormat(PFNRTSTROUTPUT pfnOutput, void *pvArgOutput, PFNSTRFORMAT pfnFormat, void *pvArgFormat, const char *pszFormat, ...)
873{
874 size_t cch;
875 va_list args;
876 va_start(args, pszFormat);
877 cch = RTStrFormatV(pfnOutput, pvArgOutput, pfnFormat, pvArgFormat, pszFormat, args);
878 va_end(args);
879 return cch;
880}
881
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