(file) Return to String.cpp CVS log (file) (dir) Up to [Pegasus] / pegasus / src / Pegasus / Common

Diff for /pegasus/src/Pegasus/Common/String.cpp between version 1.34 and 1.111.2.3

version 1.34, 2002/04/18 23:33:17 version 1.111.2.3, 2005/09/29 02:15:27
Line 1 
Line 1 
 //%/////////////////////////////////////////////////////////////////////////////  //%2005////////////////////////////////////////////////////////////////////////
 // //
 // Copyright (c) 2000, 2001 The Open group, BMC Software, Tivoli Systems, IBM  // Copyright (c) 2000, 2001, 2002 BMC Software; Hewlett-Packard Development
   // Company, L.P.; IBM Corp.; The Open Group; Tivoli Systems.
   // Copyright (c) 2003 BMC Software; Hewlett-Packard Development Company, L.P.;
   // IBM Corp.; EMC Corporation, The Open Group.
   // Copyright (c) 2004 BMC Software; Hewlett-Packard Development Company, L.P.;
   // IBM Corp.; EMC Corporation; VERITAS Software Corporation; The Open Group.
   // Copyright (c) 2005 Hewlett-Packard Development Company, L.P.; IBM Corp.;
   // EMC Corporation; VERITAS Software Corporation; The Open Group.
 // //
 // Permission is hereby granted, free of charge, to any person obtaining a copy // Permission is hereby granted, free of charge, to any person obtaining a copy
 // of this software and associated documentation files (the "Software"), to // of this software and associated documentation files (the "Software"), to
Line 22 
Line 29 
 // //
 // Author: Mike Brasher (mbrasher@bmc.com) // Author: Mike Brasher (mbrasher@bmc.com)
 // //
 // Modified By:  
 //  
 //%///////////////////////////////////////////////////////////////////////////// //%/////////////////////////////////////////////////////////////////////////////
  
   #include <cassert>
 #include <cctype>  
 #include "String.h"  
 #include "Exception.h"  
 #include "String.h" #include "String.h"
 #include <iostream>  #include "InternalException.h"
   #include "CommonUTF.h"
 PEGASUS_USING_STD;  #include "CharSet.h"
   
   #ifdef PEGASUS_STRING_ENABLE_ICU
   #include <unicode/ustring.h>
   #include <unicode/uchar.h>
   #endif
  
 PEGASUS_NAMESPACE_BEGIN PEGASUS_NAMESPACE_BEGIN
  
 #define PEGASUS_ARRAY_T String  //==============================================================================
 #include <Pegasus/Common/ArrayImpl.h>  //
 #undef PEGASUS_ARRAY_T  // File-scope definitions:
   //
 const String String::EMPTY;  //==============================================================================
  
 #if 0    // Apparently dead code  // Converts 16-bit characters to upper case.
 static inline void _SkipWhitespace(const Char16*& p)  inline Uint16 _to_upper(Uint16 x)
 { {
     while (*p && isspace(*p))      return (x & 0xFF00) ? x : CharSet::to_upper(x);
         p++;  
 } }
 #endif  
  
 inline Uint32 StrLen(const char* str)  // Converts 16-bit characters to lower case.
   inline Uint16 _to_lower(Uint16 x)
 { {
     if (!str)      return (x & 0xFF00) ? x : CharSet::to_lower(x);
         throw NullPointer();  
   
     return strlen(str);  
 } }
  
 inline Uint32 StrLen(const Char16* str)  // Rounds x to the next power of two (or just returns 8 if x < 8).
   static Uint32 _next_pow_2(Uint32 x)
 { {
     if (!str)      if (x < 8)
         throw NullPointer();          return 8;
  
     Uint32 n = 0;      x--;
       x |= (x >> 1);
       x |= (x >> 2);
       x |= (x >> 4);
       x |= (x >> 8);
       x |= (x >> 16);
       x++;
  
     while (*str++)      return x;
         n++;  }
  
     return n;  template<class P, class Q>
   static void _copy(P* p, const Q* q, size_t n)
   {
       // Use loop unrolling.
   
       while (n >= 8)
       {
           p[0] = q[0];
           p[1] = q[1];
           p[2] = q[2];
           p[3] = q[3];
           p[4] = q[4];
           p[5] = q[5];
           p[6] = q[6];
           p[7] = q[7];
           p += 8;
           q += 8;
           n -= 8;
 } }
  
 String::String()      while (n >= 4)
 { {
     _rep.append('\0');          p[0] = q[0];
           p[1] = q[1];
           p[2] = q[2];
           p[3] = q[3];
           p += 4;
           q += 4;
           n -= 4;
       }
   
       while (n--)
           *p++ = *q++;
 } }
  
 String::String(const String& x) : _rep(x._rep)  static Uint16* _find(const Uint16* s, size_t n, Uint16 c)
 { {
       while (n >= 4)
       {
           if (s[0] == c)
               return (Uint16*)s;
           if (s[1] == c)
               return (Uint16*)&s[1];
           if (s[2] == c)
               return (Uint16*)&s[2];
           if (s[3] == c)
               return (Uint16*)&s[3];
  
           n -= 4;
           s += 4;
 } }
  
 String::String(const String& x, Uint32 n)      if (n)
 { {
     _rep.append('\0');          if (*s == c)
     append(x.getData(), n);              return (Uint16*)s;
           s++;
           n--;
 } }
  
 String::String(const Char16* x) : _rep(x, StrLen(x) + 1)      if (n)
 { {
           if (*s == c)
               return (Uint16*)s;
           s++;
           n--;
       }
  
       if (n && *s == c)
           return (Uint16*)s;
   
       // Not found!
       return 0;
 } }
  
 String::String(const Char16* x, Uint32 n)  static int _compare(const Uint16* s1, const Uint16* s2)
 { {
     assign(x, n);      while (*s1 && *s2)
       {
           int r = *s1++ - *s2++;
   
           if (r)
               return r;
 } }
  
 String::String(const char* str)      if (*s2)
           return -1;
       else if (*s1)
           return 1;
   
       return 0;
   }
   
   static int _compare_no_utf8(const Uint16* s1, const char* s2)
 { {
     Uint32 n = ::strlen(str) + 1;      Uint16 c1;
     reserve(n);      Uint16 c2;
  
     while (n--)      do
         _rep.append(*str++);      {
           c1 = *s1++;
           c2 = *s2++;
   
           if (c1 == 0)
               return c1 - c2;
 } }
       while (c1 == c2);
  
 String::String(const char* str, Uint32 n_)      return c1 - c2;
   }
   
   static int _compare(const Uint16* s1, const Uint16* s2, size_t n)
 { {
     Uint32 n = _pegasusMin(strlen(str), n_);      // This should only be called when s1 and s2 have the same length.
     reserve(n + 1);  
  
     while (n--)      while (n-- && (*s1++ - *s2++) == 0)
         _rep.append(*str++);          ;
  
     _rep.append('\0');      return s1[-1] - s2[-1];
 } }
  
 String& String::assign(const Char16* x)  static inline void _copy(Uint16* s1, const Uint16* s2, size_t n)
 { {
     _rep.clear();      memcpy(s1, s2, n * sizeof(Uint16));
     _rep.append(x, StrLen(x) + 1);  
     return *this;  
 } }
  
 String& String::assign(const Char16* str, Uint32 n)  void String_throw_out_of_bounds()
 { {
     _rep.clear();      throw IndexOutOfBoundsException();
     Uint32 m = _pegasusMin(StrLen(str), n);  
     _rep.append(str, m);  
     _rep.append('\0');  
     return *this;  
 } }
  
 String& String::assign(const char* x)  #ifdef PEGASUS_STRING_NO_THROW
   # define _check_null_pointer(ARG) /* empty */
   #else
   template<class T>
   inline void _check_null_pointer(const T* ptr)
 { {
     _rep.clear();      if (!ptr)
     Uint32 n = strlen(x);          throw NullPointer();
     _rep.reserve(n + 1);  }
   #endif
  
     while (n--)  static size_t _copy_from_utf8(Uint16* dest, const char* src, size_t n)
         _rep.append(*x++);  {
       Uint16* p = dest;
       const Uint8* q = (const Uint8*)src;
  
     _rep.append('\0');      // Process leading 7-bit ASCII characters (to avoid UTF8 overhead below
       // this loop). Use factor-four loop-unrolling.
  
     return *this;      while (n >= 4 && q[0] < 128 && q[1] < 128 && q[2] < 128 && q[3] < 128)
       {
           p[0] = q[0];
           p[1] = q[1];
           p[2] = q[2];
           p[3] = q[3];
           p += 4;
           q += 4;
           n -= 4;
 } }
  
 String& String::assign(const char* x, Uint32 n_)      switch (n)
       {
           case 0:
               return p - dest;
           case 1:
               if (q[0] < 128)
 { {
     _rep.clear();                  p[0] = q[0];
                   return p + 1 - dest;
               }
               break;
           case 2:
               if (q[0] < 128 && q[1] < 128)
               {
                   p[0] = q[0];
                   p[1] = q[1];
                   return p + 2 - dest;
               }
               break;
           case 3:
               if (q[0] < 128 && q[1] < 128 && q[2] < 128)
               {
                   p[0] = q[0];
                   p[1] = q[1];
                   p[2] = q[2];
                   return p + 3 - dest;
               }
               break;
       }
  
     Uint32 n = _pegasusMin(strlen(x), n_);      // Process remaining characters.
     _rep.reserve(n + 1);  
  
     while (n--)      while (n)
         _rep.append(*x++);      {
           // Optimize for 7-bit ASCII case.
   
           if (*q < 128)
           {
               *p++ = *q++;
               n--;
           }
           else
           {
               Uint8 c = UTF_8_COUNT_TRAIL_BYTES(*q) + 1;
  
     _rep.append('\0');              if (c > n || !isValid_U8(q, c) ||
                   UTF8toUTF16(&q, q + c, &p, p + n) != 0)
               {
                   throw Exception("Bad UTF8 encoding");
               }
  
     return *this;              n -= c;
           }
 } }
  
 char* String::allocateCString(Uint32 extraBytes, Boolean noThrow) const      return p - dest;
   }
   
   // Note: dest must be at least three times src (plus an extra byte for
   // terminator).
   static inline size_t _copy_to_utf8(char* dest, const Uint16* src, size_t n)
 { {
     Uint32 n = size() + 1;      const Uint16* q = src;
     char* str = new char[n + extraBytes];      Uint8* p = (Uint8*)dest;
     char* p = str;  
     const Char16* q = getData();  
  
     for (Uint32 i = 0; i < n; i++)      while (n >= 4 && q[0] < 128 && q[1] < 128 && q[2] < 128 && q[3] < 128)
     {     {
         Uint16 c = *q++;          p[0] = q[0];
         *p++ = char(c);          p[1] = q[1];
           p[2] = q[2];
           p[3] = q[3];
           p += 4;
           q += 4;
           n -= 4;
       }
  
         if ((c & 0xff00) && !noThrow)      switch (n)
             throw TruncatedCharacter();      {
           case 0:
               return p - (Uint8*)dest;
           case 1:
               if (q[0] < 128)
               {
                   p[0] = q[0];
                   return p + 1 - (Uint8*)dest;
     }     }
               break;
           case 2:
               if (q[0] < 128 && q[1] < 128)
               {
                   p[0] = q[0];
                   p[1] = q[1];
                   return p + 2 - (Uint8*)dest;
               }
               break;
           case 3:
               if (q[0] < 128 && q[1] < 128 && q[2] < 128)
               {
                   p[0] = q[0];
                   p[1] = q[1];
                   p[2] = q[2];
                   return p + 3 - (Uint8*)dest;
               }
               break;
       }
   
       // If this line was reached, there must be characters greater than 128.
   
       UTF16toUTF8(&q, q + n, &p, p + 3 * n);
  
     return str;      return p - (Uint8*)dest;
 } }
  
 void String::appendToCString(  static inline size_t _convert(Uint16* p, const char* q, size_t n)
     char* str,  
     Uint32 length,  
     Boolean noThrow) const  
 { {
     if (!str)  #ifdef PEGASUS_STRING_NO_UTF8
         throw NullPointer();      _copy(p, q, n);
       return n;
   #else
       return _copy_from_utf8(p, q, n);
   #endif
   }
  
     Uint32 n = _pegasusMin(size(), length);  //==============================================================================
   //
   // class CString
   //
   //==============================================================================
  
     char* p = str + strlen(str);  CString::CString(const CString& cstr) : _rep(0)
     const Char16* q = getData();  {
       if (cstr._rep)
       {
           size_t n = strlen(cstr._rep) + 1;
           _rep = (char*)operator new(n);
           memcpy(_rep, cstr._rep, n);
       }
   }
  
     for (Uint32 i = 0; i < n; i++)  CString& CString::operator=(const CString& cstr)
     {     {
         Uint16 c = *q++;      if (&cstr != this)
         *p++ = char(c);      {
           if (_rep)
           {
               operator delete(_rep);
               _rep = 0;
           }
  
         if ((c & 0xff00) && !noThrow)          if (cstr._rep)
             throw TruncatedCharacter();          {
               size_t n = strlen(cstr._rep) + 1;
               _rep = (char*)operator new(n);
               memcpy(_rep, cstr._rep, n);
           }
     }     }
  
     *p = '\0';      return *this;
 } }
  
 Char16& String::operator[](Uint32 i)  //==============================================================================
   //
   // class StringRep
   //
   //==============================================================================
   
   StringRep StringRep::_empty_rep;
   
   inline StringRep* StringRep::alloc(size_t cap)
 { {
     if (i > size())      StringRep* rep = (StringRep*)::operator new(
         ThrowOutOfBounds();          sizeof(StringRep) + cap * sizeof(Uint16));
       rep->cap = cap;
       Atomic_create(&rep->refs, 1);
  
     return _rep[i];      return rep;
 } }
  
 const Char16 String::operator[](Uint32 i) const  static inline void _reserve(StringRep*& rep, Uint32 cap)
 { {
     if (i > size())      if (cap > rep->cap || Atomic_get(&rep->refs) != 1)
         ThrowOutOfBounds();      {
           size_t n = _next_pow_2(cap);
     return _rep[i];          StringRep* new_rep = StringRep::alloc(n);
           new_rep->size = rep->size;
           _copy(new_rep->data, rep->data, rep->size + 1);
           StringRep::unref(rep);
           rep = new_rep;
       }
 } }
  
 String& String::append(const Char16* str, Uint32 n)  StringRep* StringRep::create(const Uint16* data, size_t size)
 { {
     Uint32 m = _pegasusMin(StrLen(str), n);      StringRep* rep = StringRep::alloc(size);
     _rep.reserve(_rep.size() + m);      rep->size = size;
     _rep.remove(_rep.size() - 1);      _copy(rep->data, data, size);
     _rep.append(str, m);      rep->data[size] = '\0';
     _rep.append('\0');      return rep;
     return *this;  
 } }
  
 void String::remove(Uint32 pos, Uint32 size)  StringRep* StringRep::copy_on_write(StringRep* rep)
 { {
     if (size == PEG_NOT_FOUND)      // Return a new copy of rep. Release rep.
         size = this->size() - pos;  
  
     if (pos + size > this->size())      StringRep* new_rep = StringRep::alloc(rep->size);
         ThrowOutOfBounds();      new_rep->size = rep->size;
       _copy(new_rep->data, rep->data, rep->size);
       new_rep->data[new_rep->size] = '\0';
       StringRep::unref(rep);
       return new_rep;
   }
  
     if (size)  StringRep* StringRep::create(const char* data, size_t size)
         _rep.remove(pos, size);  {
       StringRep* rep = StringRep::alloc(size);
       rep->size = _convert((Uint16*)rep->data, data, size);
       rep->data[rep->size] = '\0';
   
       return rep;
 } }
  
 int String::compare(const Char16* s1, const Char16* s2, Uint32 n)  StringRep* StringRep::createAscii7(const char* data, size_t size)
 { {
     while (n--)      StringRep* rep = StringRep::alloc(size);
       _copy((Uint16*)rep->data, data, size);
       rep->data[rep->size = size] = '\0';
       return rep;
   }
   
   Uint32 StringRep::length(const Uint16* str)
     {     {
         int r = *s1++ - *s2++;      // ATTN: We could unroll this but it is infrequently called.
  
         if (r)      const Uint16* end = (Uint16*)str;
             return r;  
     }  
  
     return 0;      while (*end++)
           ;
   
       return end - str - 1;
 } }
  
 int String::compareNoCase(const char* s1, const char* s2, Uint32 n)  //==============================================================================
   //
   // class String
   //
   //==============================================================================
   
   const String String::EMPTY;
   
   String::String(const String& str, Uint32 n)
 { {
     while (n--)      _check_bounds(n, str._rep->size);
       _rep = StringRep::create(str._rep->data, n);
   }
   
   String::String(const Char16* str)
     {     {
         int r = tolower(*s1++) - tolower(*s2++);      _check_null_pointer(str);
       _rep = StringRep::create((Uint16*)str, StringRep::length((Uint16*)str));
   }
  
         if (r)  String::String(const Char16* str, Uint32 n)
             return r;  {
       _check_null_pointer(str);
       _rep = StringRep::create((Uint16*)str, n);
     }     }
  
     return 0;  String::String(const char* str)
   {
       _check_null_pointer(str);
       _rep = StringRep::create(str, strlen(str));
 } }
  
 Boolean String::equal(const String& x, const String& y)  String::String(const char* str, String::ASCII7Tag tag)
 { {
     if (x.size() != y.size())      _check_null_pointer(str);
         return false;      _rep = StringRep::createAscii7(str, strlen(str));
   }
  
     return String::compare(x.getData(), y.getData(), x.size()) == 0;  String::String(const char* str, Uint32 n)
   {
       _check_null_pointer(str);
       _rep = StringRep::create(str, n);
 } }
  
 Boolean String::equal(const String& x, const Char16* y)  String::String(const char* str, size_t n, String::ASCII7Tag tag)
 { {
     if (x.size() != StrLen(y))      _check_null_pointer(str);
         return false;      _rep = StringRep::createAscii7(str, n);
   }
  
     return String::compare(x.getData(), y, x.size()) == 0;  String::String(const String& s1, const String& s2)
   {
       size_t n1 = s1._rep->size;
       size_t n2 = s2._rep->size;
       size_t n = n1 + n2;
       _rep = StringRep::alloc(n);
       _copy(_rep->data, s1._rep->data, n1);
       _copy(_rep->data + n1, s2._rep->data, n2);
       _rep->size = n;
       _rep->data[n] = '\0';
 } }
  
 Boolean String::equal(const Char16* x, const String& y)  String::String(const String& s1, const char* s2)
 { {
     return equal(y, x);      _check_null_pointer(s2);
       size_t n1 = s1._rep->size;
       size_t n2 = strlen(s2);
       _rep = StringRep::alloc(n1 + n2);
       _copy(_rep->data, s1._rep->data, n1);
       _rep->size = n1 + _convert((Uint16*)_rep->data + n1, s2, n2);
       _rep->data[_rep->size] = '\0';
 } }
  
 Boolean String::equal(const String& x, const char* y)  String::String(const char* s1, const String& s2)
 { {
     return equal(x, String(y));      _check_null_pointer(s1);
       size_t n1 = strlen(s1);
       size_t n2 = s2._rep->size;
       _rep = StringRep::alloc(n1 + n2);
       _rep->size = n2 + _convert((Uint16*)_rep->data, s1, n1);
       _copy(_rep->data + n1, s2._rep->data, n2);
       _rep->data[_rep->size] = '\0';
 } }
  
 Boolean String::equal(const char* x, const String& y)  String& String::assign(const String& str)
   {
       if (_rep != str._rep)
 { {
     return equal(String(x), y);          StringRep::unref(_rep);
           StringRep::ref(_rep = str._rep);
 } }
  
 Boolean String::equalNoCase(const String& x, const String& y)      return *this;
   }
   
   String& String::assign(const Char16* str, Uint32 n)
 { {
     if (x.size() != y.size())      _check_null_pointer(str);
         return false;  
       if (n > _rep->cap || Atomic_get(&_rep->refs) != 1)
       {
           StringRep::unref(_rep);
           _rep = StringRep::alloc(n);
       }
  
     const Char16* p = x.getData();      _rep->size = n;
     const Char16* q = y.getData();      _copy(_rep->data, (Uint16*)str, n);
       _rep->data[n] = '\0';
  
     Uint32 n = x.size();      return *this;
   }
  
     while (n--)  String& String::assign(const char* str, Uint32 n)
     {     {
 #ifdef PEGASUS_HAS_EBCDIC      _check_null_pointer(str);
         if (*p <= 255 && *q <= 255)  
 #else      if (n > _rep->cap || Atomic_get(&_rep->refs) != 1)
         if (*p <= 127 && *q <= 127)  
 #endif  
         {         {
             if (tolower(*p++) != tolower(*q++))          StringRep::unref(_rep);
                 return false;          _rep = StringRep::alloc(n);
         }  
         else if (*p++ != *q++)  
             return false;  
     }     }
  
     return true;      _rep->size = _convert(_rep->data, str, n);
       _rep->data[_rep->size] = 0;
   
       return *this;
 } }
  
 String String::subString(Uint32 pos, Uint32 length) const  String& String::assignAscii7(const char* str, Uint32 n)
 { {
     if (pos < size())      _check_null_pointer(str);
   
       if (n > _rep->cap || Atomic_get(&_rep->refs) != 1)
     {     {
         if (length == PEG_NOT_FOUND)          StringRep::unref(_rep);
             length = size() - pos;          _rep = StringRep::alloc(n);
       }
  
         return String(getData() + pos, length);      _copy(_rep->data, str, n);
       _rep->data[_rep->size = n] = 0;
   
       return *this;
     }     }
   
   void String::clear()
   {
       if (_rep->size)
       {
           if (Atomic_get(&_rep->refs) == 1)
               _rep->size = 0;
     else     else
         return String();          {
               StringRep::unref(_rep);
               _rep = &StringRep::_empty_rep;
           }
       }
 } }
  
 Uint32 String::find(Char16 c) const  void String::reserveCapacity(Uint32 cap)
 { {
     const Char16* first = getData();      _reserve(_rep, cap);
   }
  
     for (const Char16* p = first; *p; p++)  CString String::getCString() const
     {     {
         if (*p == c)  #ifdef PEGASUS_STRING_NO_UTF8
             return  p - first;      char* str = (char*)operator new(_rep->size + 1);
       _copy(str, _rep->data, _rep->size);
       str[_rep->size] = '\0';
       return CString(str);
   #else
       Uint32 n = 3 * _rep->size;
       char* str = (char*)operator new(n + 1);
       size_t size = _copy_to_utf8(str, _rep->data, _rep->size);
       str[size] = '\0';
       return CString(str);
   #endif
     }     }
  
     return PEG_NOT_FOUND;  String& String::append(const Char16* str, Uint32 n)
   {
       _check_null_pointer(str);
   
       size_t old_size = _rep->size;
       size_t new_size = old_size + n;
       _reserve(_rep, new_size);
       _copy(_rep->data + old_size, (Uint16*)str, n);
       _rep->size = new_size;
       _rep->data[new_size] = '\0';
   
       return *this;
 } }
  
 Uint32 String::find(Uint32 pos, Char16 c) const  String& String::append(const String& str)
 { {
     const Char16* data = getData();      return append((Char16*)str._rep->data, str._rep->size);
   }
  
     for (Uint32 i = pos, n = size(); i < n; i++)  String& String::append(const char* str, Uint32 size)
     {     {
         if (data[i] == c)      _check_null_pointer(str);
             return i;  
       size_t old_size = _rep->size;
       size_t cap = old_size + size;
   
       _reserve(_rep, cap);
       _rep->size += _convert((Uint16*)_rep->data + old_size, str, size);
       _rep->data[_rep->size] = '\0';
   
       return *this;
     }     }
  
     return PEG_NOT_FOUND;  void String::remove(Uint32 index, Uint32 n)
   {
       if (n == PEG_NOT_FOUND)
           n = _rep->size - index;
   
       _check_bounds(index + n, _rep->size);
   
       if (Atomic_get(&_rep->refs) != 1)
           _rep = StringRep::copy_on_write(_rep);
   
       assert(index + n <= _rep->size);
   
       size_t rem = _rep->size - (index + n);
       Uint16* data = _rep->data;
   
       if (rem)
           memmove(data + index, data + index + n, rem * sizeof(Uint16));
   
       _rep->size -= n;
       data[_rep->size] = '\0';
 } }
  
 Uint32 String::find(const String& s) const  String String::subString(Uint32 index, Uint32 n) const
 { {
     const Char16* pSubStr = s.getData();      // ATTN: this implementation is very permissive but used for
     const Char16* pStr = getData();      // backwards compatibility.
     Uint32 subStrLen = s.size();  
     Uint32 strLen = size();  
  
     if (subStrLen > strLen)      if (index < _rep->size)
     {     {
         return PEG_NOT_FOUND;          if (n == PEG_NOT_FOUND || n > _rep->size - index)
               n = _rep->size - index;
   
           return String((Char16*)_rep->data + index, n);
     }     }
  
     // loop to find first char match      return String();
     Uint32 loc = 0;  
     for( ; loc <= (strLen-subStrLen); loc++)  
     {  
         if (*pStr++ == *pSubStr)  // match first char  
         {  
             // point to substr 2nd char  
             const Char16* p = pSubStr + 1;  
   
             // Test remaining chars for equal  
             Uint32 i = 1;  
             for (; i < subStrLen; i++)  
                 if (*pStr++ != *p++ )  
                     {pStr--; break;} // break from loop  
             if (i == subStrLen)  
                 return loc;  
         }         }
   
   Uint32 String::find(Char16 c) const
   {
       Uint16* p = (Uint16*)_find(_rep->data, _rep->size, c);
   
       if (p)
           return p - _rep->data;
   
       return PEG_NOT_FOUND;
     }     }
   
   Uint32 String::find(Uint32 index, Char16 c) const
   {
       _check_bounds(index, _rep->size);
   
       if (index >= _rep->size)
           return PEG_NOT_FOUND;
   
       Uint16* p = (Uint16*)_find(_rep->data + index, _rep->size - index, c);
   
       if (p)
           return p - _rep->data;
   
     return PEG_NOT_FOUND;     return PEG_NOT_FOUND;
 } }
  
 Uint32 String::find(const char* s) const  Uint32 String::_find_aux(const Char16* s, Uint32 n) const
 { {
     return find(String(s));      _check_null_pointer(s);
   
       const Uint16* data = _rep->data;
       size_t rem = _rep->size;
   
       while (n <= rem)
       {
           Uint16* p = (Uint16*)_find(data, rem, s[0]);
   
           if (!p)
               break;
   
           if (memcmp(p, s, n * sizeof(Uint16)) == 0)
               return p - _rep->data;
   
           p++;
           rem -= p - data;
           data = p;
 } }
  
 Uint32 String::find(const Char16* s) const      return PEG_NOT_FOUND;
   }
   
   Uint32 String::find(const char* s) const
 { {
       _check_null_pointer(s);
   
       // ATTN: implement faster version of this that doesn't create a string
       // object.
     return find(String(s));     return find(String(s));
 } }
  
 Uint32 String::reverseFind(Char16 c) const Uint32 String::reverseFind(Char16 c) const
 { {
     const Char16* first = getData();      Uint16 x = c;
     const Char16* last = getData() + size();      Uint16* p = _rep->data;
       Uint16* q = _rep->data + _rep->size;
  
     while (last != first)      while (q != p)
     {     {
         if (*--last == c)          if (*--q == x)
             return last - first;              return q - p;
     }     }
  
     return PEG_NOT_FOUND;     return PEG_NOT_FOUND;
Line 428 
Line 807 
  
 void String::toLower() void String::toLower()
 { {
     for (Char16* p = &_rep[0]; *p; p++)  #ifdef PEGASUS_STRING_ENABLE_ICU
   
       if (InitializeICU::initICUSuccessful())
       {
           //// First calculate size of resulting string. u_strToLower() returns
           //// only the size when zero is passed as the destination size argument.
   
           UErrorCode err = U_ZERO_ERROR;
   
           int32_t new_size = u_strToLower(
               NULL, 0, (UChar*)_rep->data, _rep->size, NULL, &err);
   
           //// Reserve enough space for the result.
   
           if ((Uint32)new_size > _rep->cap)
               _reserve(_rep, new_size);
   
           //// Perform the conversion (overlapping buffers are allowed).
   
           u_strToLower((UChar*)_rep->data, new_size,
               (UChar*)_rep->data, _rep->size, NULL, &err);
   
           _rep->size = new_size;
       }
   
   #endif /* PEGASUS_STRING_ENABLE_ICU */
   
       if (Atomic_get(&_rep->refs) != 1)
           _rep = StringRep::copy_on_write(_rep);
   
       Uint16* p = _rep->data;
       size_t n = _rep->size;
   
       for (; n--; p++)
     {     {
 #ifdef PEGASUS_HAS_EBCDIC          if (!(*p & 0xFF00))
         if (*p <= 255)              *p = _to_lower(*p);
 #else  
         if (*p <= 127)  
 #endif  
             *p = tolower(*p);  
     }     }
 } }
  
 void String::translate(Char16 fromChar, Char16 toChar)  void String::toUpper()
 { {
     for (Char16* p = &_rep[0]; *p; p++)  #ifdef PEGASUS_STRING_ENABLE_ICU
   
       if (InitializeICU::initICUSuccessful())
     {     {
         if (*p == fromChar)          //// First calculate size of resulting string. u_strToUpper() returns
             *p = toChar;          //// only the size when zero is passed as the destination size argument.
   
           UErrorCode err = U_ZERO_ERROR;
   
           int32_t new_size = u_strToUpper(
               NULL, 0, (UChar*)_rep->data, _rep->size, NULL, &err);
   
           //// Reserve enough space for the result.
   
           if ((Uint32)new_size > _rep->cap)
               _reserve(_rep, new_size);
   
           //// Perform the conversion (overlapping buffers are allowed).
   
           u_strToUpper((UChar*)_rep->data, new_size,
               (UChar*)_rep->data, _rep->size, NULL, &err);
   
           _rep->size = new_size;
       }
   
   #endif /* PEGASUS_STRING_ENABLE_ICU */
   
       if (Atomic_get(&_rep->refs) != 1)
           _rep = StringRep::copy_on_write(_rep);
   
       Uint16* p = _rep->data;
       size_t n = _rep->size;
   
       for (; n--; p++)
           *p = _to_upper(*p);
     }     }
   
   int String::compare(const String& s1, const String& s2, Uint32 n)
   {
       assert(n <= s1._rep->size);
       assert(n <= s2._rep->size);
   
       // Ignoring error in which n is greater than s1.size() or s2.size()
       return _compare(s1._rep->data, s2._rep->data, n);
 } }
  
 int String::compare(const Char16* s1, const Char16* s2)  int String::compare(const String& s1, const String& s2)
 { {
     while (*s1 && *s2)      return _compare(s1._rep->data, s2._rep->data);
   }
   
   int String::compare(const String& s1, const char* s2)
     {     {
         int r = *s1++ - *s2++;      _check_null_pointer(s2);
  
         if (r)  #ifdef PEGASUS_STRING_NO_UTF8
             return r;      return _compare_no_utf8(s1._rep->data, s2);
   #else
       // ATTN: optimize this!
       return String::compare(s1, String(s2));
   #endif
     }     }
  
     if (*s2)  int String::compareNoCase(const String& str1, const String& str2)
         return -1;  {
     else if (*s1)  #ifdef PEGASUS_STRING_ENABLE_ICU
         return 1;  
  
     return 0;      if (InitializeICU::initICUSuccessful())
       {
           return  u_strcasecmp(
               str1._rep->data, str2._rep->data, U_FOLD_CASE_DEFAULT);
 } }
  
 int String::compareNoCase(const char* s1, const char* s2)  #endif /* PEGASUS_STRING_ENABLE_ICU */
 {  
       const Uint16* s1 = str1._rep->data;
       const Uint16* s2 = str2._rep->data;
   
     while (*s1 && *s2)     while (*s1 && *s2)
     {     {
         int r = tolower(*s1++) - tolower(*s2++);          int r = _to_lower(*s1++) - _to_lower(*s2++);
  
         if (r)         if (r)
             return r;             return r;
Line 484 
Line 943 
     return 0;     return 0;
 } }
  
 PEGASUS_STD(ostream)& operator<<(PEGASUS_STD(ostream)& os, const String& x)  Boolean String::equalNoCase_aux(const String& s1, const String& s2)
 { {
     for (Uint32 i = 0, n = x.size(); i < n; i++)  #ifdef PEGASUS_STRING_ENABLE_ICU
         os << x[i];  
  
     return os;      return String::compareNoCase(s1, s2) == 0;
 }  
   #else /* PEGASUS_STRING_ENABLE_ICU */
  
 void String::toLower(char* str)      Uint16* p = (Uint16*)s1._rep->data;
       Uint16* q = (Uint16*)s2._rep->data;
       Uint32 n = s2._rep->size;
   
       while (n >= 8)
       {
           if (((p[0] - q[0]) && (_to_upper(p[0]) - _to_upper(q[0]))) ||
               ((p[1] - q[1]) && (_to_upper(p[1]) - _to_upper(q[1]))) ||
               ((p[2] - q[2]) && (_to_upper(p[2]) - _to_upper(q[2]))) ||
               ((p[3] - q[3]) && (_to_upper(p[3]) - _to_upper(q[3]))) ||
               ((p[4] - q[4]) && (_to_upper(p[4]) - _to_upper(q[4]))) ||
               ((p[5] - q[5]) && (_to_upper(p[5]) - _to_upper(q[5]))) ||
               ((p[6] - q[6]) && (_to_upper(p[6]) - _to_upper(q[6]))) ||
               ((p[7] - q[7]) && (_to_upper(p[7]) - _to_upper(q[7]))))
 { {
     while (*str)              return false;
         tolower(*str++);          }
   
           n -= 8;
           p += 8;
           q += 8;
 } }
  
 String ToLower(const String& str)      while (n >= 4)
 { {
     String tmp(str);          if (((p[0] - q[0]) && (_to_upper(p[0]) - _to_upper(q[0]))) ||
               ((p[1] - q[1]) && (_to_upper(p[1]) - _to_upper(q[1]))) ||
               ((p[2] - q[2]) && (_to_upper(p[2]) - _to_upper(q[2]))) ||
               ((p[3] - q[3]) && (_to_upper(p[3]) - _to_upper(q[3]))))
           {
               return false;
           }
  
     for (Uint32 i = 0, n = tmp.size(); i < n; i++)          n -= 4;
           p += 4;
           q += 4;
       }
   
       while (n--)
     {     {
         Char16 c = tmp[i];          if (((p[0] - q[0]) && (_to_upper(p[0]) - _to_upper(q[0]))))
               return false;
  
 #ifdef PEGASUS_HAS_EBCDIC          p++;
         if (c <= 255)          q++;
 #else  
         if (c <= 127)  
 #endif  
             tmp[i] = tolower(c);  
     }     }
  
     return tmp;      return true;
   
   #endif /* PEGASUS_STRING_ENABLE_ICU */
 } }
  
 int CompareNoCase(const char* s1, const char* s2)  Boolean String::equalNoCase(const String& s1, const char* s2)
 { {
     while (*s1 && *s2)      _check_null_pointer(s2);
   
   #if defined(PEGASUS_STRING_ENABLE_ICU)
   
       return String::equalNoCase(s1, String(s2));
   
   #elif defined(PEGASUS_STRING_NO_UTF8)
   
       const Uint16* p1 = (Uint16*)s1._rep->data;
       const char* p2 = s2;
       size_t n = s1._rep->size;
   
       while (n--)
     {     {
         int r = tolower(*s1++) - tolower(*s2++);          if (!*p2)
               return false;
  
         if (r)          if (_to_upper(*p1++) != CharSet::to_upper(int(*p2++)))
             return r;              return false;
     }     }
  
     if (*s2)      return true;
         return -1;  
     else if (*s1)  
         return 1;  
  
     return 0;  #else /* PEGASUS_STRING_ENABLE_ICU */
   
       // ATTN: optimize this!
       return String::equalNoCase(s1, String(s2));
   
   #endif /* PEGASUS_STRING_ENABLE_ICU */
 } }
  
 Boolean GetLine(PEGASUS_STD(istream)& is, String& line)  Boolean String::equal(const String& s1, const String& s2)
 { {
     line.clear();      return s1._rep->size == s2._rep->size && memcmp(s1._rep->data,
           s2._rep->data, s1._rep->size * sizeof(Uint16)) == 0;
     Boolean gotChar = false;  }
     char c;  
  
     while (is.get(c))  Boolean String::equal(const String& s1, const char* s2)
     {     {
         gotChar = true;  #ifdef PEGASUS_STRING_NO_UTF8
  
         if (c == '\n')      _check_null_pointer(s2);
             break;  
  
         line.append(c);      const Uint16* p = (Uint16*)s1._rep->data;
       const char* q = s2;
   
       while (*p && *q)
       {
           if (*p++ != Uint16(*q++))
               return false;
     }     }
  
     return gotChar;      return !(*p || *q);
   
   #else /* PEGASUS_STRING_NO_UTF8 */
   
       // ATTN: optimize this!
       return String::equal(s1, String(s2));
   
   #endif /* PEGASUS_STRING_NO_UTF8 */
 } }
  
 String::~String()  PEGASUS_STD(ostream)& operator<<(PEGASUS_STD(ostream)& os, const String& str)
 { {
 }  #if defined(PEGASUS_OS_OS400)
   
       CString cstr = str.getCString();
       const char* utf8str = cstr;
       os << utf8str;
   
   #elif defined(PEGASUS_STRING_ENABLE_ICU)
  
 String& String::assign(const String& x)      if (InitializeICU::initICUSuccessful())
 { {
     _rep = x._rep;          char *buf = NULL;
     return *this;          const int size = str.size() * 6;
           UnicodeString UniStr(
               (const UChar *)str.getChar16Data(), (int32_t)str.size());
           Uint32 bufsize = UniStr.extract(0,size,buf);
           buf = new char[bufsize+1];
           UniStr.extract(0,bufsize,buf);
           os << buf;
           os.flush();
           delete [] buf;
 } }
  
 String& String::append(const Char16& c)  #endif /* PEGASUS_OS_OS400 */
   
       for (Uint32 i = 0, n = str.size(); i < n; i++)
 { {
     _rep.insert(_rep.size() - 1, c);          Uint16 code = str[i];
     return *this;  
           if (code > 0 && !(code & 0xFF00))
               os << char(code);
           else
           {
               // Print in hex format:
               char buffer[8];
               sprintf(buffer, "\\x%04X", code);
               os << buffer;
           }
 } }
  
 void String::clear()      return os;
   }
   
   void String::_append_char_aux()
   {
       StringRep* tmp;
   
       if (_rep->cap)
 { {
     _rep.clear();          tmp = StringRep::alloc(2 * _rep->cap);
     _rep.append('\0');          tmp->size = _rep->size;
           _copy(tmp->data, _rep->data, _rep->size);
 } }
       else
       {
           tmp = StringRep::alloc(8);
           tmp->size = 0;
       }
   
       StringRep::unref(_rep);
       _rep = tmp;
   }
   
   PEGASUS_NAMESPACE_END
   
   /*
   ================================================================================
   
   String optimizations:
   
       1.  Added mechanism allowing certain functions to be inlined only when
           used by internal Pegasus modules. External modules (i.e., providers)
           link to a non-inline version, which allows for binary compatibility.
   
       2.  Implemented copy-on-write with atomic increment/decrement. This
           yieled a 10% improvement for the 'gc' benchmark and a 11% improvment
           for the 'ni1000' benchmark.
   
       3.  Employed loop unrolling in several places. For example, see:
  
 void String::print() const              static Uint16* _find(const Uint16* s, size_t n, Uint16 c);
   
       4.  Used the "empty-rep" optimization (described in whitepaper from the
           GCC Developers Summit). This reduced default construction to a simple
           pointer assignment.
   
               inline String::String() : _rep(&_empty_rep) { }
   
       5.  Implemented Uint16 versions of toupper() and tolower() using tables.
           For example:
   
               static const char _upper[] =
               {
                   0,1,2,...255
               };
   
               inline Uint16 _to_upper(Uint16 x)
 { {
     cout << *this << endl;                  return (x & 0xFF00) ? x : _upper[x];
 } }
  
 void String::reserve(Uint32 capacity)          This outperforms the system implementation by avoiding an anding
           operation.
   
       6.  Implemented char* version of the following member functions to
           eliminate unecessary creation of anonymous string objects
           (temporaries).
   
               String(const String& s1, const char* s2);
               String(const char* s1, const String& s2);
               String& String::operator=(const char* str);
               Uint32 String::find(const char* s) const;
               bool String::equal(const String& s1, const char* s2);
               static int String::compare(const String& s1, const char* s2);
               String& String::append(const char* str);
               String& String::append(const char* str, Uint32 size);
               static bool String::equalNoCase(const String& s1, const char* s2);
               String& operator=(const char* str)
               String& String::assign(const char* str)
               String& String::append(const char* str)
               Boolean operator==(const String& s1, const char* s2)
               Boolean operator==(const char* s1, const String& s2)
               Boolean operator!=(const String& s1, const char* s2)
               Boolean operator!=(const char* s1, const String& s2)
               Boolean operator<(const String& s1, const char* s2)
               Boolean operator<(const char* s1, const String& s2)
               Boolean operator>(const String& s1, const char* s2)
               Boolean operator>(const char* s1, const String& s2)
               Boolean operator<=(const String& s1, const char* s2)
               Boolean operator<=(const char* s1, const String& s2)
               Boolean operator>=(const String& s1, const char* s2)
               Boolean operator>=(const char* s1, const String& s2)
               String operator+(const String& s1, const char* s2)
               String operator+(const char* s1, const String& s2)
   
       7.  Optimized _next_pow_2(), used in rounding the capacity to the next
           power of two (algorithm from the book "Hacker's Delight").
   
               static Uint32 _next_pow_2(Uint32 x)
 { {
     _rep.reserve(capacity + 1);                  if (x < 8)
                       return 8;
   
                   x--;
                   x |= (x >> 1);
                   x |= (x >> 2);
                   x |= (x >> 4);
                   x |= (x >> 8);
                   x |= (x >> 16);
                   x++;
   
                   return x;
 } }
  
 const Array<String>& EmptyStringArray()      8.  Implemented "concatenating constructors" to eliminate temporaries
           created by operator+(). This scheme employs the "return-value
           optimization" described by Stan Lippman.
   
               inline String operator+(const String& s1, const String& s2)
 { {
     static Array<String> tmp;                  return String(s1, s2, 0);
     return tmp;  
 } }
  
 PEGASUS_NAMESPACE_END      9.  Experimented to find the optimial initial size for a short string.
           Eight seems to offer the best tradoff between space and time.
   
       10. Inlined all members of the Char16 class.
   
       11. Used Uint16 internally in the String class. This showed no improvememnt
           since Char16 was already fully inlined and was essentially reduced to
           Uint16 in any case.
   
       12. Implemented conditional logic (#if) allowing error checking logic to
           be excluded to better performance. Examples include bounds checking
           and null-pointer checking.
   
       13. Used memcpy() and memcmp() where possible. These are implemented using
           the rep family of intructions under Intel and are much faster.
   
       14. Used loop unrolling, jump-tables, and short-circuiting to reduce UTF8
           copy routine overhead.
   
       15. Added ASCII7 form of the constructor and assign().
   
               String s("hello world", String::ASCII7);
   
               s.assignAscii7("hello world");
   
           This avoids checking for UTF8 when it is not needed.
   
   ================================================================================
   */


Legend:
Removed from v.1.34  
changed lines
  Added in v.1.111.2.3

No CVS admin address has been configured
Powered by
ViewCVS 0.9.2