]>
Commit | Line | Data |
---|---|---|
6fe7ccc8 | 1 | /* |
ed1e77d3 | 2 | * Copyright (C) 2012-2015 Apple Inc. All rights reserved. |
6fe7ccc8 A |
3 | * |
4 | * Redistribution and use in source and binary forms, with or without | |
5 | * modification, are permitted provided that the following conditions | |
6 | * are met: | |
7 | * 1. Redistributions of source code must retain the above copyright | |
8 | * notice, this list of conditions and the following disclaimer. | |
9 | * 2. Redistributions in binary form must reproduce the above copyright | |
10 | * notice, this list of conditions and the following disclaimer in the | |
11 | * documentation and/or other materials provided with the distribution. | |
12 | * | |
13 | * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY | |
14 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
15 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | |
16 | * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR | |
17 | * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | |
18 | * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | |
19 | * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | |
20 | * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | |
21 | * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |
22 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
23 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
24 | */ | |
25 | ||
26 | #include "config.h" | |
27 | #include "JSStringJoiner.h" | |
28 | ||
81345200 | 29 | #include "JSCInlines.h" |
6fe7ccc8 | 30 | |
6fe7ccc8 A |
31 | namespace JSC { |
32 | ||
ed1e77d3 A |
33 | template<typename CharacterType> |
34 | static inline void appendStringToData(CharacterType*& data, StringView string) | |
6fe7ccc8 | 35 | { |
ed1e77d3 A |
36 | string.getCharactersWithUpconvert(data); |
37 | data += string.length(); | |
38 | } | |
6fe7ccc8 | 39 | |
ed1e77d3 A |
40 | template<typename CharacterType> |
41 | static inline String joinStrings(const Vector<StringViewWithUnderlyingString>& strings, StringView separator, unsigned joinedLength) | |
42 | { | |
43 | ASSERT(joinedLength); | |
6fe7ccc8 | 44 | |
ed1e77d3 A |
45 | CharacterType* data; |
46 | String result = StringImpl::tryCreateUninitialized(joinedLength, data); | |
47 | if (result.isNull()) | |
48 | return result; | |
49 | ||
50 | appendStringToData(data, strings[0].view); | |
51 | ||
52 | unsigned size = strings.size(); | |
53 | ||
54 | switch (separator.length()) { | |
55 | case 0: | |
56 | for (unsigned i = 1; i < size; ++i) | |
57 | appendStringToData(data, strings[i].view); | |
58 | break; | |
59 | case 1: { | |
60 | CharacterType separatorCharacter = separator[0]; | |
61 | for (unsigned i = 1; i < size; ++i) { | |
62 | *data++ = separatorCharacter; | |
63 | appendStringToData(data, strings[i].view); | |
6fe7ccc8 | 64 | } |
ed1e77d3 A |
65 | break; |
66 | } | |
67 | default: | |
68 | for (unsigned i = 1; i < size; ++i) { | |
69 | appendStringToData(data, separator); | |
70 | appendStringToData(data, strings[i].view); | |
6fe7ccc8 A |
71 | } |
72 | } | |
ed1e77d3 | 73 | ASSERT(data == result.characters<CharacterType>() + joinedLength); |
6fe7ccc8 | 74 | |
ed1e77d3 | 75 | return result; |
6fe7ccc8 A |
76 | } |
77 | ||
ed1e77d3 | 78 | inline unsigned JSStringJoiner::joinedLength(ExecState& state) const |
6fe7ccc8 | 79 | { |
ed1e77d3 A |
80 | unsigned numberOfStrings = m_strings.size(); |
81 | if (!numberOfStrings) | |
82 | return 0; | |
6fe7ccc8 | 83 | |
ed1e77d3 A |
84 | Checked<unsigned, RecordOverflow> separatorLength = m_separator.length(); |
85 | Checked<unsigned, RecordOverflow> totalSeparatorsLength = separatorLength * (numberOfStrings - 1); | |
86 | Checked<unsigned, RecordOverflow> totalLength = totalSeparatorsLength + m_accumulatedStringsLength; | |
6fe7ccc8 | 87 | |
ed1e77d3 A |
88 | unsigned result; |
89 | if (totalLength.safeGet(result) == CheckedState::DidOverflow) { | |
90 | throwOutOfMemoryError(&state); | |
91 | return 0; | |
6fe7ccc8 | 92 | } |
ed1e77d3 | 93 | return result; |
6fe7ccc8 A |
94 | } |
95 | ||
ed1e77d3 | 96 | JSValue JSStringJoiner::join(ExecState& state) |
6fe7ccc8 | 97 | { |
ed1e77d3 | 98 | ASSERT(m_strings.size() <= m_strings.capacity()); |
6fe7ccc8 | 99 | |
ed1e77d3 A |
100 | unsigned length = joinedLength(state); |
101 | if (state.hadException()) | |
102 | return jsUndefined(); | |
6fe7ccc8 | 103 | |
ed1e77d3 A |
104 | if (!length) |
105 | return jsEmptyString(&state); | |
106 | ||
107 | String result; | |
108 | if (m_isAll8Bit) | |
109 | result = joinStrings<LChar>(m_strings, m_separator, length); | |
6fe7ccc8 | 110 | else |
ed1e77d3 | 111 | result = joinStrings<UChar>(m_strings, m_separator, length); |
6fe7ccc8 | 112 | |
ed1e77d3 A |
113 | if (result.isNull()) |
114 | return throwOutOfMemoryError(&state); | |
6fe7ccc8 | 115 | |
ed1e77d3 | 116 | return jsString(&state, WTF::move(result)); |
6fe7ccc8 A |
117 | } |
118 | ||
119 | } |