]> git.saurik.com Git - apple/javascriptcore.git/blob - wtf/text/WTFString.cpp
842d755c8ee665539e320feaf86bfeb931a556ea
[apple/javascriptcore.git] / wtf / text / WTFString.cpp
1 /*
2 * (C) 1999 Lars Knoll (knoll@kde.org)
3 * Copyright (C) 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved.
4 * Copyright (C) 2007-2009 Torch Mobile, Inc.
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Library General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Library General Public License for more details.
15 *
16 * You should have received a copy of the GNU Library General Public License
17 * along with this library; see the file COPYING.LIB. If not, write to
18 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19 * Boston, MA 02110-1301, USA.
20 */
21
22 #include "config.h"
23 #include "WTFString.h"
24
25 #include <limits>
26 #include <stdarg.h>
27 #include <wtf/ASCIICType.h>
28 #include <wtf/text/CString.h>
29 #include <wtf/StringExtras.h>
30 #include <wtf/Vector.h>
31 #include <wtf/dtoa.h>
32 #include <wtf/unicode/UTF8.h>
33 #include <wtf/unicode/Unicode.h>
34
35 using namespace WTF;
36 using namespace WTF::Unicode;
37
38 namespace WebCore {
39
40 String::String(const UChar* str)
41 {
42 if (!str)
43 return;
44
45 int len = 0;
46 while (str[len] != UChar(0))
47 len++;
48
49 m_impl = StringImpl::create(str, len);
50 }
51
52 void String::append(const String& str)
53 {
54 if (str.isEmpty())
55 return;
56
57 // FIXME: This is extremely inefficient. So much so that we might want to take this
58 // out of String's API. We can make it better by optimizing the case where exactly
59 // one String is pointing at this StringImpl, but even then it's going to require a
60 // call to fastMalloc every single time.
61 if (str.m_impl) {
62 if (m_impl) {
63 UChar* data;
64 RefPtr<StringImpl> newImpl =
65 StringImpl::createUninitialized(m_impl->length() + str.length(), data);
66 memcpy(data, m_impl->characters(), m_impl->length() * sizeof(UChar));
67 memcpy(data + m_impl->length(), str.characters(), str.length() * sizeof(UChar));
68 m_impl = newImpl.release();
69 } else
70 m_impl = str.m_impl;
71 }
72 }
73
74 void String::append(char c)
75 {
76 // FIXME: This is extremely inefficient. So much so that we might want to take this
77 // out of String's API. We can make it better by optimizing the case where exactly
78 // one String is pointing at this StringImpl, but even then it's going to require a
79 // call to fastMalloc every single time.
80 if (m_impl) {
81 UChar* data;
82 RefPtr<StringImpl> newImpl =
83 StringImpl::createUninitialized(m_impl->length() + 1, data);
84 memcpy(data, m_impl->characters(), m_impl->length() * sizeof(UChar));
85 data[m_impl->length()] = c;
86 m_impl = newImpl.release();
87 } else
88 m_impl = StringImpl::create(&c, 1);
89 }
90
91 void String::append(UChar c)
92 {
93 // FIXME: This is extremely inefficient. So much so that we might want to take this
94 // out of String's API. We can make it better by optimizing the case where exactly
95 // one String is pointing at this StringImpl, but even then it's going to require a
96 // call to fastMalloc every single time.
97 if (m_impl) {
98 UChar* data;
99 RefPtr<StringImpl> newImpl =
100 StringImpl::createUninitialized(m_impl->length() + 1, data);
101 memcpy(data, m_impl->characters(), m_impl->length() * sizeof(UChar));
102 data[m_impl->length()] = c;
103 m_impl = newImpl.release();
104 } else
105 m_impl = StringImpl::create(&c, 1);
106 }
107
108 String operator+(const String& a, const String& b)
109 {
110 if (a.isEmpty())
111 return b;
112 if (b.isEmpty())
113 return a;
114 String c = a;
115 c += b;
116 return c;
117 }
118
119 String operator+(const String& s, const char* cs)
120 {
121 return s + String(cs);
122 }
123
124 String operator+(const char* cs, const String& s)
125 {
126 return String(cs) + s;
127 }
128
129 void String::insert(const String& str, unsigned pos)
130 {
131 if (str.isEmpty()) {
132 if (str.isNull())
133 return;
134 if (isNull())
135 m_impl = str.impl();
136 return;
137 }
138 insert(str.characters(), str.length(), pos);
139 }
140
141 void String::append(const UChar* charactersToAppend, unsigned lengthToAppend)
142 {
143 if (!m_impl) {
144 if (!charactersToAppend)
145 return;
146 m_impl = StringImpl::create(charactersToAppend, lengthToAppend);
147 return;
148 }
149
150 if (!lengthToAppend)
151 return;
152
153 ASSERT(charactersToAppend);
154 UChar* data;
155 RefPtr<StringImpl> newImpl =
156 StringImpl::createUninitialized(length() + lengthToAppend, data);
157 memcpy(data, characters(), length() * sizeof(UChar));
158 memcpy(data + length(), charactersToAppend, lengthToAppend * sizeof(UChar));
159 m_impl = newImpl.release();
160 }
161
162 void String::insert(const UChar* charactersToInsert, unsigned lengthToInsert, unsigned position)
163 {
164 if (position >= length()) {
165 append(charactersToInsert, lengthToInsert);
166 return;
167 }
168
169 ASSERT(m_impl);
170
171 if (!lengthToInsert)
172 return;
173
174 ASSERT(charactersToInsert);
175 UChar* data;
176 RefPtr<StringImpl> newImpl =
177 StringImpl::createUninitialized(length() + lengthToInsert, data);
178 memcpy(data, characters(), position * sizeof(UChar));
179 memcpy(data + position, charactersToInsert, lengthToInsert * sizeof(UChar));
180 memcpy(data + position + lengthToInsert, characters() + position, (length() - position) * sizeof(UChar));
181 m_impl = newImpl.release();
182 }
183
184 UChar32 String::characterStartingAt(unsigned i) const
185 {
186 if (!m_impl || i >= m_impl->length())
187 return 0;
188 return m_impl->characterStartingAt(i);
189 }
190
191 void String::truncate(unsigned position)
192 {
193 if (position >= length())
194 return;
195 UChar* data;
196 RefPtr<StringImpl> newImpl = StringImpl::createUninitialized(position, data);
197 memcpy(data, characters(), position * sizeof(UChar));
198 m_impl = newImpl.release();
199 }
200
201 void String::remove(unsigned position, int lengthToRemove)
202 {
203 if (lengthToRemove <= 0)
204 return;
205 if (position >= length())
206 return;
207 if (static_cast<unsigned>(lengthToRemove) > length() - position)
208 lengthToRemove = length() - position;
209 UChar* data;
210 RefPtr<StringImpl> newImpl =
211 StringImpl::createUninitialized(length() - lengthToRemove, data);
212 memcpy(data, characters(), position * sizeof(UChar));
213 memcpy(data + position, characters() + position + lengthToRemove,
214 (length() - lengthToRemove - position) * sizeof(UChar));
215 m_impl = newImpl.release();
216 }
217
218 String String::substring(unsigned pos, unsigned len) const
219 {
220 if (!m_impl)
221 return String();
222 return m_impl->substring(pos, len);
223 }
224
225 String String::lower() const
226 {
227 if (!m_impl)
228 return String();
229 return m_impl->lower();
230 }
231
232 String String::upper() const
233 {
234 if (!m_impl)
235 return String();
236 return m_impl->upper();
237 }
238
239 String String::stripWhiteSpace() const
240 {
241 if (!m_impl)
242 return String();
243 return m_impl->stripWhiteSpace();
244 }
245
246 String String::simplifyWhiteSpace() const
247 {
248 if (!m_impl)
249 return String();
250 return m_impl->simplifyWhiteSpace();
251 }
252
253 String String::removeCharacters(CharacterMatchFunctionPtr findMatch) const
254 {
255 if (!m_impl)
256 return String();
257 return m_impl->removeCharacters(findMatch);
258 }
259
260 String String::foldCase() const
261 {
262 if (!m_impl)
263 return String();
264 return m_impl->foldCase();
265 }
266
267 bool String::percentage(int& result) const
268 {
269 if (!m_impl || !m_impl->length())
270 return false;
271
272 if ((*m_impl)[m_impl->length() - 1] != '%')
273 return false;
274
275 result = charactersToIntStrict(m_impl->characters(), m_impl->length() - 1);
276 return true;
277 }
278
279 const UChar* String::charactersWithNullTermination()
280 {
281 if (!m_impl)
282 return 0;
283 if (m_impl->hasTerminatingNullCharacter())
284 return m_impl->characters();
285 m_impl = StringImpl::createWithTerminatingNullCharacter(*m_impl);
286 return m_impl->characters();
287 }
288
289 String String::format(const char *format, ...)
290 {
291 #if PLATFORM(QT)
292 // Use QString::vsprintf to avoid the locale dependent formatting of vsnprintf.
293 // https://bugs.webkit.org/show_bug.cgi?id=18994
294 va_list args;
295 va_start(args, format);
296
297 QString buffer;
298 buffer.vsprintf(format, args);
299
300 va_end(args);
301
302 return buffer;
303
304 #elif OS(WINCE)
305 va_list args;
306 va_start(args, format);
307
308 Vector<char, 256> buffer;
309
310 int bufferSize = 256;
311 buffer.resize(bufferSize);
312 for (;;) {
313 int written = vsnprintf(buffer.data(), bufferSize, format, args);
314 va_end(args);
315
316 if (written == 0)
317 return String("");
318 if (written > 0)
319 return StringImpl::create(buffer.data(), written);
320
321 bufferSize <<= 1;
322 buffer.resize(bufferSize);
323 va_start(args, format);
324 }
325
326 #else
327 va_list args;
328 va_start(args, format);
329
330 Vector<char, 256> buffer;
331
332 // Do the format once to get the length.
333 #if COMPILER(MSVC)
334 int result = _vscprintf(format, args);
335 #else
336 char ch;
337 int result = vsnprintf(&ch, 1, format, args);
338 // We need to call va_end() and then va_start() again here, as the
339 // contents of args is undefined after the call to vsnprintf
340 // according to http://man.cx/snprintf(3)
341 //
342 // Not calling va_end/va_start here happens to work on lots of
343 // systems, but fails e.g. on 64bit Linux.
344 va_end(args);
345 va_start(args, format);
346 #endif
347
348 if (result == 0)
349 return String("");
350 if (result < 0)
351 return String();
352 unsigned len = result;
353 buffer.grow(len + 1);
354
355 // Now do the formatting again, guaranteed to fit.
356 vsnprintf(buffer.data(), buffer.size(), format, args);
357
358 va_end(args);
359
360 return StringImpl::create(buffer.data(), len);
361 #endif
362 }
363
364 String String::number(short n)
365 {
366 return String::format("%hd", n);
367 }
368
369 String String::number(unsigned short n)
370 {
371 return String::format("%hu", n);
372 }
373
374 String String::number(int n)
375 {
376 return String::format("%d", n);
377 }
378
379 String String::number(unsigned n)
380 {
381 return String::format("%u", n);
382 }
383
384 String String::number(long n)
385 {
386 return String::format("%ld", n);
387 }
388
389 String String::number(unsigned long n)
390 {
391 return String::format("%lu", n);
392 }
393
394 String String::number(long long n)
395 {
396 #if OS(WINDOWS) && !PLATFORM(QT)
397 return String::format("%I64i", n);
398 #else
399 return String::format("%lli", n);
400 #endif
401 }
402
403 String String::number(unsigned long long n)
404 {
405 #if OS(WINDOWS) && !PLATFORM(QT)
406 return String::format("%I64u", n);
407 #else
408 return String::format("%llu", n);
409 #endif
410 }
411
412 String String::number(double n)
413 {
414 return String::format("%.6lg", n);
415 }
416
417 int String::toIntStrict(bool* ok, int base) const
418 {
419 if (!m_impl) {
420 if (ok)
421 *ok = false;
422 return 0;
423 }
424 return m_impl->toIntStrict(ok, base);
425 }
426
427 unsigned String::toUIntStrict(bool* ok, int base) const
428 {
429 if (!m_impl) {
430 if (ok)
431 *ok = false;
432 return 0;
433 }
434 return m_impl->toUIntStrict(ok, base);
435 }
436
437 int64_t String::toInt64Strict(bool* ok, int base) const
438 {
439 if (!m_impl) {
440 if (ok)
441 *ok = false;
442 return 0;
443 }
444 return m_impl->toInt64Strict(ok, base);
445 }
446
447 uint64_t String::toUInt64Strict(bool* ok, int base) const
448 {
449 if (!m_impl) {
450 if (ok)
451 *ok = false;
452 return 0;
453 }
454 return m_impl->toUInt64Strict(ok, base);
455 }
456
457 intptr_t String::toIntPtrStrict(bool* ok, int base) const
458 {
459 if (!m_impl) {
460 if (ok)
461 *ok = false;
462 return 0;
463 }
464 return m_impl->toIntPtrStrict(ok, base);
465 }
466
467
468 int String::toInt(bool* ok) const
469 {
470 if (!m_impl) {
471 if (ok)
472 *ok = false;
473 return 0;
474 }
475 return m_impl->toInt(ok);
476 }
477
478 unsigned String::toUInt(bool* ok) const
479 {
480 if (!m_impl) {
481 if (ok)
482 *ok = false;
483 return 0;
484 }
485 return m_impl->toUInt(ok);
486 }
487
488 int64_t String::toInt64(bool* ok) const
489 {
490 if (!m_impl) {
491 if (ok)
492 *ok = false;
493 return 0;
494 }
495 return m_impl->toInt64(ok);
496 }
497
498 uint64_t String::toUInt64(bool* ok) const
499 {
500 if (!m_impl) {
501 if (ok)
502 *ok = false;
503 return 0;
504 }
505 return m_impl->toUInt64(ok);
506 }
507
508 intptr_t String::toIntPtr(bool* ok) const
509 {
510 if (!m_impl) {
511 if (ok)
512 *ok = false;
513 return 0;
514 }
515 return m_impl->toIntPtr(ok);
516 }
517
518 double String::toDouble(bool* ok) const
519 {
520 if (!m_impl) {
521 if (ok)
522 *ok = false;
523 return 0.0;
524 }
525 return m_impl->toDouble(ok);
526 }
527
528 float String::toFloat(bool* ok) const
529 {
530 if (!m_impl) {
531 if (ok)
532 *ok = false;
533 return 0.0f;
534 }
535 return m_impl->toFloat(ok);
536 }
537
538 String String::threadsafeCopy() const
539 {
540 if (!m_impl)
541 return String();
542 return m_impl->threadsafeCopy();
543 }
544
545 String String::crossThreadString() const
546 {
547 if (!m_impl)
548 return String();
549 return m_impl->crossThreadString();
550 }
551
552 void String::split(const String& separator, bool allowEmptyEntries, Vector<String>& result) const
553 {
554 result.clear();
555
556 int startPos = 0;
557 int endPos;
558 while ((endPos = find(separator, startPos)) != -1) {
559 if (allowEmptyEntries || startPos != endPos)
560 result.append(substring(startPos, endPos - startPos));
561 startPos = endPos + separator.length();
562 }
563 if (allowEmptyEntries || startPos != static_cast<int>(length()))
564 result.append(substring(startPos));
565 }
566
567 void String::split(const String& separator, Vector<String>& result) const
568 {
569 return split(separator, false, result);
570 }
571
572 void String::split(UChar separator, bool allowEmptyEntries, Vector<String>& result) const
573 {
574 result.clear();
575
576 int startPos = 0;
577 int endPos;
578 while ((endPos = find(separator, startPos)) != -1) {
579 if (allowEmptyEntries || startPos != endPos)
580 result.append(substring(startPos, endPos - startPos));
581 startPos = endPos + 1;
582 }
583 if (allowEmptyEntries || startPos != static_cast<int>(length()))
584 result.append(substring(startPos));
585 }
586
587 void String::split(UChar separator, Vector<String>& result) const
588 {
589 return split(String(&separator, 1), false, result);
590 }
591
592 Vector<char> String::ascii() const
593 {
594 if (m_impl)
595 return m_impl->ascii();
596
597 const char* nullMsg = "(null impl)";
598 Vector<char, 2048> buffer;
599 for (int i = 0; nullMsg[i]; ++i)
600 buffer.append(nullMsg[i]);
601
602 buffer.append('\0');
603 return buffer;
604 }
605
606 CString String::latin1() const
607 {
608 // Basic Latin1 (ISO) encoding - Unicode characters 0..255 are
609 // preserved, characters outside of this range are converted to '?'.
610
611 unsigned length = this->length();
612 const UChar* characters = this->characters();
613
614 char* characterBuffer;
615 CString result = CString::newUninitialized(length, characterBuffer);
616
617 for (unsigned i = 0; i < length; ++i) {
618 UChar ch = characters[i];
619 characterBuffer[i] = ch > 255 ? '?' : ch;
620 }
621
622 return result;
623 }
624
625 // Helper to write a three-byte UTF-8 code point to the buffer, caller must check room is available.
626 static inline void putUTF8Triple(char*& buffer, UChar ch)
627 {
628 ASSERT(ch >= 0x0800);
629 *buffer++ = static_cast<char>(((ch >> 12) & 0x0F) | 0xE0);
630 *buffer++ = static_cast<char>(((ch >> 6) & 0x3F) | 0x80);
631 *buffer++ = static_cast<char>((ch & 0x3F) | 0x80);
632 }
633
634 CString String::utf8() const
635 {
636 unsigned length = this->length();
637 const UChar* characters = this->characters();
638
639 // Allocate a buffer big enough to hold all the characters
640 // (an individual UTF-16 UChar can only expand to 3 UTF-8 bytes).
641 // Optimization ideas, if we find this function is hot:
642 // * We could speculatively create a CStringBuffer to contain 'length'
643 // characters, and resize if necessary (i.e. if the buffer contains
644 // non-ascii characters). (Alternatively, scan the buffer first for
645 // ascii characters, so we know this will be sufficient).
646 // * We could allocate a CStringBuffer with an appropriate size to
647 // have a good chance of being able to write the string into the
648 // buffer without reallocing (say, 1.5 x length).
649 Vector<char, 1024> bufferVector(length * 3);
650
651 char* buffer = bufferVector.data();
652 ConversionResult result = convertUTF16ToUTF8(&characters, characters + length, &buffer, buffer + bufferVector.size(), false);
653 ASSERT(result != sourceIllegal); // Only produced from strict conversion.
654 ASSERT(result != targetExhausted); // (length * 3) should be sufficient for any conversion
655
656 // If a high surrogate is left unconverted, treat it the same was as an unpaired high surrogate
657 // would have been handled in the middle of a string with non-strict conversion - which is to say,
658 // simply encode it to UTF-8.
659 if (result == sourceExhausted) {
660 // This should be one unpaired high surrogate.
661 ASSERT((characters + 1) == (characters + length));
662 ASSERT((*characters >= 0xD800) && (*characters <= 0xDBFF));
663 // There should be room left, since one UChar hasn't been converted.
664 ASSERT((buffer + 3) <= (buffer + bufferVector.size()));
665 putUTF8Triple(buffer, *characters);
666 }
667
668 return CString(bufferVector.data(), buffer - bufferVector.data());
669 }
670
671 String String::fromUTF8(const char* stringStart, size_t length)
672 {
673 if (!stringStart)
674 return String();
675
676 // We'll use a StringImpl as a buffer; if the source string only contains ascii this should be
677 // the right length, if there are any multi-byte sequences this buffer will be too large.
678 UChar* buffer;
679 String stringBuffer(StringImpl::createUninitialized(length, buffer));
680 UChar* bufferEnd = buffer + length;
681
682 // Try converting into the buffer.
683 const char* stringCurrent = stringStart;
684 if (convertUTF8ToUTF16(&stringCurrent, stringStart + length, &buffer, bufferEnd) != conversionOK)
685 return String();
686
687 // stringBuffer is full (the input must have been all ascii) so just return it!
688 if (buffer == bufferEnd)
689 return stringBuffer;
690
691 // stringBuffer served its purpose as a buffer, copy the contents out into a new string.
692 unsigned utf16Length = buffer - stringBuffer.characters();
693 ASSERT(utf16Length < length);
694 return String(stringBuffer.characters(), utf16Length);
695 }
696
697 String String::fromUTF8(const char* string)
698 {
699 if (!string)
700 return String();
701 return fromUTF8(string, strlen(string));
702 }
703
704 String String::fromUTF8WithLatin1Fallback(const char* string, size_t size)
705 {
706 String utf8 = fromUTF8(string, size);
707 if (!utf8)
708 return String(string, size);
709 return utf8;
710 }
711
712 // String Operations
713
714 static bool isCharacterAllowedInBase(UChar c, int base)
715 {
716 if (c > 0x7F)
717 return false;
718 if (isASCIIDigit(c))
719 return c - '0' < base;
720 if (isASCIIAlpha(c)) {
721 if (base > 36)
722 base = 36;
723 return (c >= 'a' && c < 'a' + base - 10)
724 || (c >= 'A' && c < 'A' + base - 10);
725 }
726 return false;
727 }
728
729 template <typename IntegralType>
730 static inline IntegralType toIntegralType(const UChar* data, size_t length, bool* ok, int base)
731 {
732 static const IntegralType integralMax = std::numeric_limits<IntegralType>::max();
733 static const bool isSigned = std::numeric_limits<IntegralType>::is_signed;
734 const IntegralType maxMultiplier = integralMax / base;
735
736 IntegralType value = 0;
737 bool isOk = false;
738 bool isNegative = false;
739
740 if (!data)
741 goto bye;
742
743 // skip leading whitespace
744 while (length && isSpaceOrNewline(*data)) {
745 length--;
746 data++;
747 }
748
749 if (isSigned && length && *data == '-') {
750 length--;
751 data++;
752 isNegative = true;
753 } else if (length && *data == '+') {
754 length--;
755 data++;
756 }
757
758 if (!length || !isCharacterAllowedInBase(*data, base))
759 goto bye;
760
761 while (length && isCharacterAllowedInBase(*data, base)) {
762 length--;
763 IntegralType digitValue;
764 UChar c = *data;
765 if (isASCIIDigit(c))
766 digitValue = c - '0';
767 else if (c >= 'a')
768 digitValue = c - 'a' + 10;
769 else
770 digitValue = c - 'A' + 10;
771
772 if (value > maxMultiplier || (value == maxMultiplier && digitValue > (integralMax % base) + isNegative))
773 goto bye;
774
775 value = base * value + digitValue;
776 data++;
777 }
778
779 #if COMPILER(MSVC)
780 #pragma warning(push, 0)
781 #pragma warning(disable:4146)
782 #endif
783
784 if (isNegative)
785 value = -value;
786
787 #if COMPILER(MSVC)
788 #pragma warning(pop)
789 #endif
790
791 // skip trailing space
792 while (length && isSpaceOrNewline(*data)) {
793 length--;
794 data++;
795 }
796
797 if (!length)
798 isOk = true;
799 bye:
800 if (ok)
801 *ok = isOk;
802 return isOk ? value : 0;
803 }
804
805 static unsigned lengthOfCharactersAsInteger(const UChar* data, size_t length)
806 {
807 size_t i = 0;
808
809 // Allow leading spaces.
810 for (; i != length; ++i) {
811 if (!isSpaceOrNewline(data[i]))
812 break;
813 }
814
815 // Allow sign.
816 if (i != length && (data[i] == '+' || data[i] == '-'))
817 ++i;
818
819 // Allow digits.
820 for (; i != length; ++i) {
821 if (!isASCIIDigit(data[i]))
822 break;
823 }
824
825 return i;
826 }
827
828 int charactersToIntStrict(const UChar* data, size_t length, bool* ok, int base)
829 {
830 return toIntegralType<int>(data, length, ok, base);
831 }
832
833 unsigned charactersToUIntStrict(const UChar* data, size_t length, bool* ok, int base)
834 {
835 return toIntegralType<unsigned>(data, length, ok, base);
836 }
837
838 int64_t charactersToInt64Strict(const UChar* data, size_t length, bool* ok, int base)
839 {
840 return toIntegralType<int64_t>(data, length, ok, base);
841 }
842
843 uint64_t charactersToUInt64Strict(const UChar* data, size_t length, bool* ok, int base)
844 {
845 return toIntegralType<uint64_t>(data, length, ok, base);
846 }
847
848 intptr_t charactersToIntPtrStrict(const UChar* data, size_t length, bool* ok, int base)
849 {
850 return toIntegralType<intptr_t>(data, length, ok, base);
851 }
852
853 int charactersToInt(const UChar* data, size_t length, bool* ok)
854 {
855 return toIntegralType<int>(data, lengthOfCharactersAsInteger(data, length), ok, 10);
856 }
857
858 unsigned charactersToUInt(const UChar* data, size_t length, bool* ok)
859 {
860 return toIntegralType<unsigned>(data, lengthOfCharactersAsInteger(data, length), ok, 10);
861 }
862
863 int64_t charactersToInt64(const UChar* data, size_t length, bool* ok)
864 {
865 return toIntegralType<int64_t>(data, lengthOfCharactersAsInteger(data, length), ok, 10);
866 }
867
868 uint64_t charactersToUInt64(const UChar* data, size_t length, bool* ok)
869 {
870 return toIntegralType<uint64_t>(data, lengthOfCharactersAsInteger(data, length), ok, 10);
871 }
872
873 intptr_t charactersToIntPtr(const UChar* data, size_t length, bool* ok)
874 {
875 return toIntegralType<intptr_t>(data, lengthOfCharactersAsInteger(data, length), ok, 10);
876 }
877
878 double charactersToDouble(const UChar* data, size_t length, bool* ok)
879 {
880 if (!length) {
881 if (ok)
882 *ok = false;
883 return 0.0;
884 }
885
886 Vector<char, 256> bytes(length + 1);
887 for (unsigned i = 0; i < length; ++i)
888 bytes[i] = data[i] < 0x7F ? data[i] : '?';
889 bytes[length] = '\0';
890 char* end;
891 double val = WTF::strtod(bytes.data(), &end);
892 if (ok)
893 *ok = (end == 0 || *end == '\0');
894 return val;
895 }
896
897 float charactersToFloat(const UChar* data, size_t length, bool* ok)
898 {
899 // FIXME: This will return ok even when the string fits into a double but not a float.
900 return static_cast<float>(charactersToDouble(data, length, ok));
901 }
902
903 } // namespace WebCore
904
905 #ifndef NDEBUG
906 // For use in the debugger - leaks memory
907 WebCore::String* string(const char*);
908
909 WebCore::String* string(const char* s)
910 {
911 return new WebCore::String(s);
912 }
913 #endif