VirtualBox

source: vbox/trunk/include/VBox/com/string.h@ 23907

Last change on this file since 23907 was 23907, checked in by vboxsync, 15 years ago

Main: fix memory leak

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 18.7 KB
Line 
1/* $Id: string.h 23907 2009-10-20 15:27:40Z vboxsync $ */
2
3/** @file
4 * MS COM / XPCOM Abstraction Layer:
5 * Smart string classes declaration
6 */
7
8/*
9 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * The contents of this file may alternatively be used under the terms
20 * of the Common Development and Distribution License Version 1.0
21 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
22 * VirtualBox OSE distribution, in which case the provisions of the
23 * CDDL are applicable instead of those of the GPL.
24 *
25 * You may elect to license modified versions of this file under the
26 * terms and conditions of either the GPL or the CDDL or both.
27 *
28 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
29 * Clara, CA 95054 USA or visit http://www.sun.com if you need
30 * additional information or have any questions.
31 */
32
33#ifndef ___VBox_com_string_h
34#define ___VBox_com_string_h
35
36/* Make sure all the stdint.h macros are included - must come first! */
37#ifndef __STDC_LIMIT_MACROS
38# define __STDC_LIMIT_MACROS
39#endif
40#ifndef __STDC_CONSTANT_MACROS
41# define __STDC_CONSTANT_MACROS
42#endif
43
44#if defined (VBOX_WITH_XPCOM)
45# include <nsMemory.h>
46#endif
47
48#include "VBox/com/defs.h"
49#include "VBox/com/assert.h"
50
51#include <iprt/cpputils.h>
52#include <iprt/alloc.h>
53#include <iprt/ministring_cpp.h>
54
55namespace com
56{
57
58class Utf8Str;
59
60/**
61 * Helper class that represents the |BSTR| type and hides platform-specific
62 * implementation details.
63 *
64 * This class uses COM/XPCOM-provided memory management routines to allocate
65 * and free string buffers. This makes it possible to:
66 * - use it as a type of member variables of COM/XPCOM components and pass
67 * their values to callers through component methods' output parameters
68 * using the #cloneTo() operation;
69 * - adopt (take ownership of) string buffers returned in output parameters
70 * of COM methods using the #asOutParam() operation and correctly free them
71 * afterwards.
72 */
73class Bstr
74{
75public:
76
77 typedef BSTR String;
78 typedef CBSTR ConstString;
79
80 Bstr () : bstr (NULL) {}
81
82 Bstr (const Bstr &that) : bstr (NULL) { raw_copy (bstr, that.bstr); }
83 Bstr (CBSTR that) : bstr (NULL) { raw_copy (bstr, that); }
84
85#if defined (VBOX_WITH_XPCOM)
86 Bstr (const wchar_t *that) : bstr (NULL)
87 {
88 AssertCompile (sizeof (wchar_t) == sizeof (OLECHAR));
89 raw_copy (bstr, (CBSTR) that);
90 }
91#endif
92
93 Bstr (const iprt::MiniString &that);
94 Bstr (const char *that);
95
96 /** Shortcut that calls #alloc(aSize) right after object creation. */
97 Bstr (size_t aSize) : bstr (NULL) { alloc (aSize); }
98
99 ~Bstr () { setNull(); }
100
101 Bstr &operator = (const Bstr &that) { safe_assign (that.bstr); return *this; }
102 Bstr &operator = (CBSTR that) { safe_assign (that); return *this; }
103
104 Bstr &operator = (const Utf8Str &that);
105 Bstr &operator = (const char *that);
106
107 Bstr &setNull()
108 {
109 if (bstr)
110 {
111 ::SysFreeString (bstr);
112 bstr = NULL;
113 }
114 return *this;
115 }
116
117 Bstr &setNullIfEmpty()
118 {
119 if (bstr && *bstr == 0)
120 {
121 ::SysFreeString (bstr);
122 bstr = NULL;
123 }
124 return *this;
125 }
126
127 /**
128 * Allocates memory for a string capable to store \a aSize - 1 characters;
129 * in other words, aSize includes the terminating zero character. If \a aSize
130 * is zero, or if a memory allocation error occurs, this object will become null.
131 */
132 Bstr &alloc (size_t aSize)
133 {
134 setNull();
135 if (aSize)
136 {
137 unsigned int size = (unsigned int) aSize; Assert (size == aSize);
138 bstr = ::SysAllocStringLen (NULL, size - 1);
139 if (bstr)
140 bstr [0] = 0;
141 }
142 return *this;
143 }
144
145 int compare (CBSTR str) const
146 {
147 return ::RTUtf16Cmp ((PRTUTF16) bstr, (PRTUTF16) str);
148 }
149
150 int compare (BSTR str) const
151 {
152 return ::RTUtf16Cmp ((PRTUTF16) bstr, (PRTUTF16) str);
153 }
154
155 bool operator==(const Bstr &that) const { return !compare (that.bstr); }
156 bool operator!=(const Bstr &that) const { return !!compare (that.bstr); }
157 bool operator==(CBSTR that) const { return !compare (that); }
158 bool operator==(BSTR that) const { return !compare (that); }
159
160#if defined (VBOX_WITH_XPCOM)
161 bool operator!=(const wchar_t *that) const
162 {
163 AssertCompile (sizeof (wchar_t) == sizeof (OLECHAR));
164 return !!compare ((CBSTR) that);
165 }
166 bool operator==(const wchar_t *that) const
167 {
168 AssertCompile (sizeof (wchar_t) == sizeof (OLECHAR));
169 return !compare ((CBSTR) that);
170 }
171#endif
172
173 bool operator!=(CBSTR that) const { return !!compare (that); }
174 bool operator!=(BSTR that) const { return !!compare (that); }
175 bool operator<(const Bstr &that) const { return compare (that.bstr) < 0; }
176 bool operator<(CBSTR that) const { return compare (that) < 0; }
177 bool operator<(BSTR that) const { return compare (that) < 0; }
178#if defined (VBOX_WITH_XPCOM)
179 bool operator<(const wchar_t *that) const
180 {
181 AssertCompile (sizeof (wchar_t) == sizeof (OLECHAR));
182 return compare ((CBSTR) that) < 0;
183 }
184#endif
185
186 int compareIgnoreCase (CBSTR str) const
187 {
188 return ::RTUtf16LocaleICmp (bstr, str);
189 }
190
191 bool isNull() const { return bstr == NULL; }
192 operator bool() const { return !isNull(); }
193
194 bool isEmpty() const { return isNull() || *bstr == 0; }
195
196 size_t length() const { return isNull() ? 0 : ::RTUtf16Len ((PRTUTF16) bstr); }
197
198 /** Intended to to pass instances as |CBSTR| input parameters to methods. */
199 operator CBSTR () const { return bstr; }
200
201 /**
202 * Intended to to pass instances as |BSTR| input parameters to methods.
203 * Note that we have to provide this mutable BSTR operator since in MS COM
204 * input BSTR parameters of interface methods are not const.
205 */
206 operator BSTR () { return bstr; }
207
208 /**
209 * The same as operator CBSTR(), but for situations where the compiler
210 * cannot typecast implicitly (for example, in printf() argument list).
211 */
212 CBSTR raw() const { return bstr; }
213
214 /**
215 * Returns a non-const raw pointer that allows to modify the string directly.
216 * @warning
217 * Be sure not to modify data beyond the allocated memory! The
218 * guaranteed size of the allocated memory is at least #length()
219 * bytes after creation and after every assignment operation.
220 */
221 BSTR mutableRaw() { return bstr; }
222
223 /**
224 * Intended to assign copies of instances to |BSTR| out parameters from
225 * within the interface method. Transfers the ownership of the duplicated
226 * string to the caller.
227 */
228 const Bstr &cloneTo (BSTR *pstr) const
229 {
230 if (pstr)
231 {
232 if (*pstr)
233 {
234 ::SysFreeString (*pstr);
235 *pstr = NULL;
236 }
237 raw_copy (*pstr, bstr);
238 }
239 return *this;
240 }
241
242 /**
243 * Intended to assign instances to |BSTR| out parameters from within the
244 * interface method. Transfers the ownership of the original string to the
245 * caller and resets the instance to null.
246 *
247 * As opposed to cloneTo(), this method doesn't create a copy of the
248 * string.
249 */
250 Bstr &detachTo (BSTR *pstr)
251 {
252 *pstr = bstr;
253 bstr = NULL;
254 return *this;
255 }
256
257 /**
258 * Intended to assign copies of instances to |char *| out parameters from
259 * within the interface method. Transfers the ownership of the duplicated
260 * string to the caller.
261 */
262 const Bstr &cloneTo (char **pstr) const;
263
264 /**
265 * Intended to pass instances as |BSTR| out parameters to methods.
266 * Takes the ownership of the returned data.
267 */
268 BSTR *asOutParam() { setNull(); return &bstr; }
269
270 /**
271 * Static immutable null object. May be used for comparison purposes.
272 */
273 static const Bstr Null;
274
275protected:
276
277 void safe_assign (CBSTR str)
278 {
279 if (bstr != str)
280 {
281 setNull();
282 raw_copy (bstr, str);
283 }
284 }
285
286 inline static void raw_copy(BSTR &ls, CBSTR rs)
287 {
288 if (rs)
289 ls = ::SysAllocString((const OLECHAR *) rs);
290 }
291
292 inline static void raw_copy (BSTR &ls, const char *rs)
293 {
294 if (rs)
295 {
296 PRTUTF16 s = NULL;
297 ::RTStrToUtf16 (rs, &s);
298 raw_copy (ls, (BSTR) s);
299 ::RTUtf16Free (s);
300 }
301 }
302
303 BSTR bstr;
304
305 friend class Utf8Str; /* to access our raw_copy() */
306};
307
308/* symmetric compare operators */
309inline bool operator==(CBSTR l, const Bstr &r) { return r.operator== (l); }
310inline bool operator!=(CBSTR l, const Bstr &r) { return r.operator!= (l); }
311inline bool operator==(BSTR l, const Bstr &r) { return r.operator== (l); }
312inline bool operator!=(BSTR l, const Bstr &r) { return r.operator!= (l); }
313
314////////////////////////////////////////////////////////////////////////////////
315
316/**
317 * Helper class that represents UTF8 (|char *|) strings. Useful in
318 * conjunction with Bstr to simplify conversions between UTF16 (|BSTR|)
319 * and UTF8.
320 *
321 * This class uses COM/XPCOM-provided memory management routines to allocate
322 * and free string buffers. This makes it possible to:
323 * - use it as a type of member variables of COM/XPCOM components and pass
324 * their values to callers through component methods' output parameters
325 * using the #cloneTo() operation;
326 * - adopt (take ownership of) string buffers returned in output parameters
327 * of COM methods using the #asOutParam() operation and correctly free them
328 * afterwards.
329 */
330class Utf8Str : public iprt::MiniString
331{
332public:
333
334 Utf8Str() {}
335
336 Utf8Str(const MiniString &that)
337 : MiniString(that)
338 {}
339
340 Utf8Str(const char *that)
341 : MiniString(that)
342 {}
343
344 Utf8Str(const Bstr &that)
345 {
346 copyFrom(that);
347 }
348
349 Utf8Str(CBSTR that)
350 {
351 copyFrom(that);
352 }
353
354 Utf8Str& operator=(const MiniString &that)
355 {
356 MiniString::operator=(that);
357 return *this;
358 }
359
360 Utf8Str& operator=(const char *that)
361 {
362 MiniString::operator=(that);
363 return *this;
364 }
365
366 Utf8Str& operator=(const Bstr &that)
367 {
368 cleanup();
369 copyFrom(that);
370 return *this;
371 }
372
373 Utf8Str& operator=(CBSTR that)
374 {
375 cleanup();
376 copyFrom(that);
377 return *this;
378 }
379
380 /**
381 * Intended to assign instances to |char *| out parameters from within the
382 * interface method. Transfers the ownership of the duplicated string to the
383 * caller.
384 */
385 const Utf8Str& cloneTo(char **pstr) const
386 {
387 if (pstr)
388 *pstr = RTStrDup(m_psz);
389 return *this;
390 }
391
392 /**
393 * Intended to assign instances to |char *| out parameters from within the
394 * interface method. Transfers the ownership of the original string to the
395 * caller and resets the instance to null.
396 *
397 * As opposed to cloneTo(), this method doesn't create a copy of the
398 * string.
399 */
400 Utf8Str& detachTo(char **pstr)
401 {
402 *pstr = m_psz;
403 m_psz = NULL;
404 m_cbAllocated = 0;
405 m_cbLength = 0;
406 return *this;
407 }
408
409 /**
410 * Intended to assign instances to |BSTR| out parameters from within the
411 * interface method. Transfers the ownership of the duplicated string to the
412 * caller.
413 */
414 const Utf8Str& cloneTo(BSTR *pstr) const
415 {
416 if (pstr)
417 {
418 if (*pstr)
419 {
420 ::SysFreeString (*pstr);
421 *pstr = NULL;
422 }
423 Bstr::raw_copy(*pstr, m_psz);
424 }
425 return *this;
426 }
427
428 /**
429 * Converts "this" to lower case by calling RTStrToLower().
430 * @return
431 */
432 Utf8Str& toLower();
433
434 /**
435 * Converts "this" to upper case by calling RTStrToUpper().
436 * @return
437 */
438 Utf8Str& toUpper();
439
440 /**
441 * Removes a trailing slash from the member string, if present.
442 * Calls RTPathStripTrailingSlash() without having to mess with mutableRaw().
443 */
444 void stripTrailingSlash();
445
446 /**
447 * Removes a trailing filename from the member string, if present.
448 * Calls RTPathStripFilename() without having to mess with mutableRaw().
449 */
450 void stripFilename();
451
452 /**
453 * Removes a trailing file name extension from the member string, if present.
454 * Calls RTPathStripExt() without having to mess with mutableRaw().
455 */
456 void stripExt();
457
458 /**
459 * Attempts to convert the member string into a 32-bit integer.
460 *
461 * @returns 32-bit unsigned number on success.
462 * @returns 0 on failure.
463 */
464 int toInt32() const
465 {
466 return RTStrToInt32(m_psz);
467 }
468
469 /**
470 * Attempts to convert the member string into an unsigned 32-bit integer.
471 *
472 * @returns 32-bit unsigned number on success.
473 * @returns 0 on failure.
474 */
475 int toUInt32() const
476 {
477 return RTStrToUInt32(m_psz);
478 }
479
480 /**
481 * Intended to pass instances as out (|char **|) parameters to methods.
482 * Takes the ownership of the returned data.
483 */
484 char **asOutParam()
485 {
486 cleanup();
487 return &m_psz;
488 }
489
490 /**
491 * Static immutable null object. May be used for comparison purposes.
492 */
493 static const Utf8Str Null;
494
495protected:
496
497 /**
498 * As with the ministring::copyFrom() variants, this unconditionally
499 * sets the members to a copy of the given other strings and makes
500 * no assumptions about previous contents. This can therefore be used
501 * both in copy constructors, when member variables have no defined
502 * value, and in assignments after having called cleanup().
503 *
504 * This variant converts from a UTF-16 string, most probably from
505 * a Bstr assignment.
506 *
507 * @param rs
508 */
509 void copyFrom(CBSTR s)
510 {
511 if (s)
512 {
513 RTUtf16ToUtf8((PRTUTF16)s, &m_psz);
514 m_cbLength = strlen(m_psz); // TODO optimize by using a different RTUtf* function
515 m_cbAllocated = m_cbLength + 1;
516 }
517 else
518 {
519 m_cbLength = 0;
520 m_cbAllocated = 0;
521 m_psz = NULL;
522 }
523 }
524
525 friend class Bstr; /* to access our raw_copy() */
526};
527
528// work around error C2593 of the stupid MSVC 7.x ambiguity resolver
529WORKAROUND_MSVC7_ERROR_C2593_FOR_BOOL_OP (Bstr)
530
531////////////////////////////////////////////////////////////////////////////////
532
533// inlined Bstr members that depend on Utf8Str
534
535inline Bstr::Bstr(const iprt::MiniString &that)
536 : bstr(NULL)
537{
538 raw_copy(bstr, that.c_str());
539}
540
541inline Bstr::Bstr(const char *that)
542 : bstr(NULL)
543{
544 raw_copy(bstr, that);
545}
546
547inline Bstr &Bstr::operator=(const Utf8Str &that)
548{
549 setNull();
550 raw_copy(bstr, that.c_str());
551 return *this;
552}
553inline Bstr &Bstr::operator=(const char *that)
554{
555 setNull();
556 raw_copy(bstr, that);
557 return *this;
558}
559
560inline const Bstr& Bstr::cloneTo(char **pstr) const
561{
562 if (pstr)
563 {
564 Utf8Str ustr(*this);
565 ustr.detachTo(pstr);
566 }
567 return *this;
568}
569
570////////////////////////////////////////////////////////////////////////////////
571
572/**
573 * This class is a printf-like formatter for Utf8Str strings. Its purpose is
574 * to construct Utf8Str objects from a format string and a list of arguments
575 * for the format string.
576 *
577 * The usage of this class is like the following:
578 * <code>
579 * Utf8StrFmt string ("program name = %s", argv[0]);
580 * </code>
581 */
582class Utf8StrFmt : public Utf8Str
583{
584public:
585
586 /**
587 * Constructs a new string given the format string and the list
588 * of the arguments for the format string.
589 *
590 * @param format printf-like format string (in UTF-8 encoding)
591 * @param ... list of the arguments for the format string
592 */
593 explicit Utf8StrFmt (const char *format, ...)
594 {
595 va_list args;
596 va_start (args, format);
597 init (format, args);
598 va_end (args);
599 }
600
601protected:
602
603 Utf8StrFmt() {}
604
605 void init (const char *format, va_list args);
606
607private:
608
609 static DECLCALLBACK(size_t) strOutput (void *pvArg, const char *pachChars,
610 size_t cbChars);
611};
612
613/**
614 * This class is a vprintf-like formatter for Utf8Str strings. It is
615 * identical to Utf8StrFmt except that its constructor takes a va_list
616 * argument instead of ellipsis.
617 *
618 * Note that a separate class is necessary because va_list is defined as
619 * |char *| on most platforms. For this reason, if we had two overloaded
620 * constructors in Utf8StrFmt (one taking ellipsis and another one taking
621 * va_list) then composing a constructor call using exactly two |char *|
622 * arguments would cause the compiler to use the va_list overload instead of
623 * the ellipsis one which is obviously wrong. The compiler would choose
624 * va_list because ellipsis has the lowest rank when it comes to resolving
625 * overloads, as opposed to va_list which is an exact match for |char *|.
626 */
627class Utf8StrFmtVA : public Utf8StrFmt
628{
629public:
630
631 /**
632 * Constructs a new string given the format string and the list
633 * of the arguments for the format string.
634 *
635 * @param format printf-like format string (in UTF-8 encoding)
636 * @param args list of arguments for the format string
637 */
638 Utf8StrFmtVA (const char *format, va_list args) { init (format, args); }
639};
640
641/**
642 * The BstrFmt class is a shortcut to <tt>Bstr (Utf8StrFmt (...))</tt>.
643 */
644class BstrFmt : public Bstr
645{
646public:
647
648 /**
649 * Constructs a new string given the format string and the list of the
650 * arguments for the format string.
651 *
652 * @param aFormat printf-like format string (in UTF-8 encoding).
653 * @param ... List of the arguments for the format string.
654 */
655 explicit BstrFmt (const char *aFormat, ...)
656 {
657 va_list args;
658 va_start(args, aFormat);
659 raw_copy(bstr, Utf8StrFmtVA(aFormat, args).c_str());
660 va_end (args);
661 }
662};
663
664/**
665 * The BstrFmtVA class is a shortcut to <tt>Bstr (Utf8StrFmtVA (...))</tt>.
666 */
667class BstrFmtVA : public Bstr
668{
669public:
670
671 /**
672 * Constructs a new string given the format string and the list of the
673 * arguments for the format string.
674 *
675 * @param aFormat printf-like format string (in UTF-8 encoding).
676 * @param aArgs List of arguments for the format string
677 */
678 BstrFmtVA (const char *aFormat, va_list aArgs)
679 {
680 raw_copy(bstr, Utf8StrFmtVA(aFormat, aArgs).c_str());
681 }
682};
683
684} /* namespace com */
685
686#endif /* ___VBox_com_string_h */
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette