VirtualBox

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

Last change on this file since 96407 was 96407, checked in by vboxsync, 2 years ago

scm copyright and license note update

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