1 // Copyright 2016 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "src/inspector/string-16.h"
6
7 #include <algorithm>
8 #include <cctype>
9 #include <cinttypes>
10 #include <cstdlib>
11 #include <cstring>
12 #include <limits>
13 #include <string>
14
15 #include "../../third_party/inspector_protocol/crdtp/cbor.h"
16 #include "src/base/platform/platform.h"
17 #include "src/inspector/v8-string-conversions.h"
18 #include "src/numbers/conversions.h"
19
20 namespace v8_inspector {
21
22 namespace {
23
isASCII(UChar c)24 bool isASCII(UChar c) { return !(c & ~0x7F); }
25
isSpaceOrNewLine(UChar c)26 bool isSpaceOrNewLine(UChar c) {
27 return isASCII(c) && c <= ' ' && (c == ' ' || (c <= 0xD && c >= 0x9));
28 }
29
charactersToInteger(const UChar * characters,size_t length,bool * ok=nullptr)30 int64_t charactersToInteger(const UChar* characters, size_t length,
31 bool* ok = nullptr) {
32 std::vector<char> buffer;
33 buffer.reserve(length + 1);
34 for (size_t i = 0; i < length; ++i) {
35 if (!isASCII(characters[i])) {
36 if (ok) *ok = false;
37 return 0;
38 }
39 buffer.push_back(static_cast<char>(characters[i]));
40 }
41 buffer.push_back('\0');
42
43 char* endptr;
44 int64_t result =
45 static_cast<int64_t>(std::strtoll(buffer.data(), &endptr, 10));
46 if (ok) *ok = !(*endptr);
47 return result;
48 }
49 } // namespace
50
String16(const UChar * characters,size_t size)51 String16::String16(const UChar* characters, size_t size)
52 : m_impl(characters, size) {}
53
String16(const UChar * characters)54 String16::String16(const UChar* characters) : m_impl(characters) {}
55
String16(const char * characters)56 String16::String16(const char* characters)
57 : String16(characters, std::strlen(characters)) {}
58
String16(const char * characters,size_t size)59 String16::String16(const char* characters, size_t size) {
60 m_impl.resize(size);
61 for (size_t i = 0; i < size; ++i) m_impl[i] = characters[i];
62 }
63
String16(const std::basic_string<UChar> & impl)64 String16::String16(const std::basic_string<UChar>& impl) : m_impl(impl) {}
65
String16(std::basic_string<UChar> && impl)66 String16::String16(std::basic_string<UChar>&& impl) : m_impl(impl) {}
67
68 // static
fromInteger(int number)69 String16 String16::fromInteger(int number) {
70 char arr[50];
71 v8::base::Vector<char> buffer(arr, arraysize(arr));
72 return String16(v8::internal::IntToCString(number, buffer));
73 }
74
75 // static
fromInteger(size_t number)76 String16 String16::fromInteger(size_t number) {
77 const size_t kBufferSize = 50;
78 char buffer[kBufferSize];
79 #if !defined(_WIN32) && !defined(_WIN64)
80 v8::base::OS::SNPrintF(buffer, kBufferSize, "%zu", number);
81 #else
82 v8::base::OS::SNPrintF(buffer, kBufferSize, "%Iu", number);
83 #endif
84 return String16(buffer);
85 }
86
87 // static
fromInteger64(int64_t number)88 String16 String16::fromInteger64(int64_t number) {
89 char buffer[50];
90 v8::base::OS::SNPrintF(buffer, arraysize(buffer), "%" PRId64 "", number);
91 return String16(buffer);
92 }
93
94 // static
fromDouble(double number)95 String16 String16::fromDouble(double number) {
96 char arr[50];
97 v8::base::Vector<char> buffer(arr, arraysize(arr));
98 return String16(v8::internal::DoubleToCString(number, buffer));
99 }
100
101 // static
fromDouble(double number,int precision)102 String16 String16::fromDouble(double number, int precision) {
103 std::unique_ptr<char[]> str(
104 v8::internal::DoubleToPrecisionCString(number, precision));
105 return String16(str.get());
106 }
107
toInteger64(bool * ok) const108 int64_t String16::toInteger64(bool* ok) const {
109 return charactersToInteger(characters16(), length(), ok);
110 }
111
toInteger(bool * ok) const112 int String16::toInteger(bool* ok) const {
113 int64_t result = toInteger64(ok);
114 if (ok && *ok) {
115 *ok = result <= std::numeric_limits<int>::max() &&
116 result >= std::numeric_limits<int>::min();
117 }
118 return static_cast<int>(result);
119 }
120
stripWhiteSpace() const121 String16 String16::stripWhiteSpace() const {
122 if (!length()) return String16();
123
124 size_t start = 0;
125 size_t end = length() - 1;
126
127 // skip white space from start
128 while (start <= end && isSpaceOrNewLine(characters16()[start])) ++start;
129
130 // only white space
131 if (start > end) return String16();
132
133 // skip white space from end
134 while (end && isSpaceOrNewLine(characters16()[end])) --end;
135
136 if (!start && end == length() - 1) return *this;
137 return String16(characters16() + start, end + 1 - start);
138 }
139
140 String16Builder::String16Builder() = default;
141
append(const String16 & s)142 void String16Builder::append(const String16& s) {
143 m_buffer.insert(m_buffer.end(), s.characters16(),
144 s.characters16() + s.length());
145 }
146
append(UChar c)147 void String16Builder::append(UChar c) { m_buffer.push_back(c); }
148
append(char c)149 void String16Builder::append(char c) {
150 UChar u = c;
151 m_buffer.push_back(u);
152 }
153
append(const UChar * characters,size_t length)154 void String16Builder::append(const UChar* characters, size_t length) {
155 m_buffer.insert(m_buffer.end(), characters, characters + length);
156 }
157
append(const char * characters,size_t length)158 void String16Builder::append(const char* characters, size_t length) {
159 m_buffer.insert(m_buffer.end(), characters, characters + length);
160 }
161
appendNumber(int number)162 void String16Builder::appendNumber(int number) {
163 constexpr int kBufferSize = 11;
164 char buffer[kBufferSize];
165 int chars = v8::base::OS::SNPrintF(buffer, kBufferSize, "%d", number);
166 DCHECK_LE(0, chars);
167 m_buffer.insert(m_buffer.end(), buffer, buffer + chars);
168 }
169
appendNumber(size_t number)170 void String16Builder::appendNumber(size_t number) {
171 constexpr int kBufferSize = 20;
172 char buffer[kBufferSize];
173 #if !defined(_WIN32) && !defined(_WIN64)
174 int chars = v8::base::OS::SNPrintF(buffer, kBufferSize, "%zu", number);
175 #else
176 int chars = v8::base::OS::SNPrintF(buffer, kBufferSize, "%Iu", number);
177 #endif
178 DCHECK_LE(0, chars);
179 m_buffer.insert(m_buffer.end(), buffer, buffer + chars);
180 }
181
appendUnsignedAsHex(uint64_t number)182 void String16Builder::appendUnsignedAsHex(uint64_t number) {
183 constexpr int kBufferSize = 17;
184 char buffer[kBufferSize];
185 int chars =
186 v8::base::OS::SNPrintF(buffer, kBufferSize, "%016" PRIx64, number);
187 DCHECK_LE(0, chars);
188 m_buffer.insert(m_buffer.end(), buffer, buffer + chars);
189 }
190
appendUnsignedAsHex(uint32_t number)191 void String16Builder::appendUnsignedAsHex(uint32_t number) {
192 constexpr int kBufferSize = 9;
193 char buffer[kBufferSize];
194 int chars = v8::base::OS::SNPrintF(buffer, kBufferSize, "%08" PRIx32, number);
195 DCHECK_LE(0, chars);
196 m_buffer.insert(m_buffer.end(), buffer, buffer + chars);
197 }
198
appendUnsignedAsHex(uint8_t number)199 void String16Builder::appendUnsignedAsHex(uint8_t number) {
200 constexpr int kBufferSize = 3;
201 char buffer[kBufferSize];
202 int chars = v8::base::OS::SNPrintF(buffer, kBufferSize, "%02" PRIx8, number);
203 DCHECK_LE(0, chars);
204 m_buffer.insert(m_buffer.end(), buffer, buffer + chars);
205 }
206
toString()207 String16 String16Builder::toString() {
208 return String16(m_buffer.data(), m_buffer.size());
209 }
210
reserveCapacity(size_t capacity)211 void String16Builder::reserveCapacity(size_t capacity) {
212 m_buffer.reserve(capacity);
213 }
214
fromUTF8(const char * stringStart,size_t length)215 String16 String16::fromUTF8(const char* stringStart, size_t length) {
216 return String16(UTF8ToUTF16(stringStart, length));
217 }
218
fromUTF16LE(const UChar * stringStart,size_t length)219 String16 String16::fromUTF16LE(const UChar* stringStart, size_t length) {
220 #ifdef V8_TARGET_BIG_ENDIAN
221 // Need to flip the byte order on big endian machines.
222 String16Builder builder;
223 builder.reserveCapacity(length);
224 for (size_t i = 0; i < length; i++) {
225 const UChar utf16be_char =
226 stringStart[i] << 8 | (stringStart[i] >> 8 & 0x00FF);
227 builder.append(utf16be_char);
228 }
229 return builder.toString();
230 #else
231 // No need to do anything on little endian machines.
232 return String16(stringStart, length);
233 #endif // V8_TARGET_BIG_ENDIAN
234 }
235
utf8() const236 std::string String16::utf8() const {
237 return UTF16ToUTF8(m_impl.data(), m_impl.size());
238 }
239
240 } // namespace v8_inspector
241