1 /*
2 * Copyright (c) 2021 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "quickjs_native_string.h"
17
18 #include "securec.h"
19 #include "utils/log.h"
20
QuickJSNativeString(QuickJSNativeEngine * engine,JSValue value)21 QuickJSNativeString::QuickJSNativeString(QuickJSNativeEngine* engine, JSValue value) : QuickJSNativeValue(engine, value)
22 {
23 }
24
QuickJSNativeString(QuickJSNativeEngine * engine,JSAtom value)25 QuickJSNativeString::QuickJSNativeString(QuickJSNativeEngine* engine, JSAtom value)
26 : QuickJSNativeString(engine, JS_AtomToString(engine->GetContext(), value))
27 {
28 }
29
QuickJSNativeString(QuickJSNativeEngine * engine,const char * value,size_t length)30 QuickJSNativeString::QuickJSNativeString(QuickJSNativeEngine* engine, const char* value, size_t length)
31 : QuickJSNativeString(engine, JS_NewStringLen(engine->GetContext(), value, length))
32 {
33 }
34
QuickJSNativeString(QuickJSNativeEngine * engine,const char16_t * value,size_t length)35 QuickJSNativeString::QuickJSNativeString(QuickJSNativeEngine* engine, const char16_t* value, size_t length)
36 : QuickJSNativeString(
37 engine, JS_NewString16(engine->GetContext(), reinterpret_cast<const uint16_t*>(value), length))
38 {
39 }
40
~QuickJSNativeString()41 QuickJSNativeString::~QuickJSNativeString() {}
42
GetInterface(int interfaceId)43 void* QuickJSNativeString::GetInterface(int interfaceId)
44 {
45 return (NativeString::INTERFACE_ID == interfaceId) ? (NativeString*)this : nullptr;
46 }
47
GetCString(char * buffer,size_t size,size_t * length)48 void QuickJSNativeString::GetCString(char* buffer, size_t size, size_t* length)
49 {
50 const char* str = JS_ToCStringLen(engine_->GetContext(), length, value_);
51
52 if (str == nullptr) {
53 HILOG_ERROR("JS_ToCStringLen return value is null");
54 return;
55 }
56
57 if (buffer != nullptr && length != nullptr) {
58 int ret = strncpy_s(buffer, size, str, *length);
59 if (ret != EOK) {
60 HILOG_ERROR("strncpy_s failed");
61 }
62 }
63 JS_FreeCString(engine_->GetContext(), str);
64 }
GetCString16(char16_t * buffer,size_t size,size_t * length)65 void QuickJSNativeString::GetCString16(char16_t* buffer, size_t size, size_t* length)
66 {
67 const char* str = JS_ToCStringLen(engine_->GetContext(), length, value_);
68 if (str == nullptr) {
69 HILOG_ERROR("JS_ToCStringLen return value is null");
70 return;
71 }
72 if (length != nullptr) {
73 auto ret = Utf8ToUtf16Length(str, strlen(str));
74 if (ret == -1) {
75 HILOG_ERROR("JS_ToCStringLen length is invalid");
76 return;
77 }
78 *length = ret;
79 if (buffer != nullptr) {
80 memset_s(buffer, sizeof(char16_t) * size, 0x0, sizeof(char16_t) * size);
81 Utf8ToUtf16(str, strlen(str), buffer, size);
82 }
83 }
84 JS_FreeCString(engine_->GetContext(), str);
85 }
86
GetLength()87 size_t QuickJSNativeString::GetLength()
88 {
89 size_t length = 0;
90 GetCString(nullptr, 0, &length);
91 return length;
92 }
93
EncodeWriteUtf8(char * buffer,size_t bufferSize,int32_t * nchars)94 size_t QuickJSNativeString::EncodeWriteUtf8(char* buffer, size_t bufferSize, int32_t* nchars)
95 {
96 if (buffer == nullptr || nchars == nullptr) {
97 HILOG_ERROR("buffer is null or nchars is null");
98 return 0;
99 }
100
101 JSContext* ctx = engine_->GetContext();
102 JSValue lengthVal = JS_GetPropertyStr(ctx, value_, "length");
103 if (JS_IsException(lengthVal)) {
104 HILOG_ERROR("Failed to obtain the length");
105 return 0;
106 }
107 uint32_t len = 0;
108 if (JS_ToUint32(ctx, &len, lengthVal)) {
109 JS_FreeValue(ctx, lengthVal);
110 HILOG_ERROR("Cannot convert length to uint32_t");
111 return 0;
112 }
113
114 size_t pos = 0;
115 size_t writableSize = bufferSize;
116 uint32_t i = 0;
117 for (; i < len; i++) {
118 JSValue ch = JS_GetPropertyUint32(ctx, value_, i);
119 size_t chLen = 0;
120 const char* str = JS_ToCStringLen(ctx, &chLen, ch);
121 JS_FreeValue(ctx, ch);
122 if (chLen > writableSize) {
123 JS_FreeCString(ctx, str);
124 break;
125 }
126
127 int ret = memcpy_s((buffer + pos), writableSize, str, chLen);
128 if (ret != EOK) {
129 HILOG_ERROR("memcpy_s failed");
130 JS_FreeCString(ctx, str);
131 break;
132 }
133
134 writableSize -= chLen;
135 pos = bufferSize - writableSize;
136 JS_FreeCString(ctx, str);
137 }
138
139 *nchars = i;
140 JS_FreeValue(ctx, lengthVal);
141 return pos;
142 }
143
Utf8ToUtf16(const char * utf8Str,size_t u8len,char16_t * u16str,size_t u16len)144 char16_t* QuickJSNativeString::Utf8ToUtf16(const char* utf8Str, size_t u8len, char16_t* u16str, size_t u16len)
145 {
146 if (u16len == 0) {
147 return u16str;
148 }
149 const char* u8end = utf8Str + u8len;
150 const char* u8cur = utf8Str;
151 const char16_t* u16end = u16str + u16len - 1;
152 constexpr uint8_t offset = 10;
153 char16_t* u16cur = u16str;
154
155 while ((u8cur < u8end) && (u16cur < u16end)) {
156 size_t len = Utf8CodePointLen(*u8cur);
157 uint32_t codepoint = Utf8ToUtf32CodePoint(u8cur, len);
158 // Convert the UTF32 codepoint to one or more UTF16 codepoints
159 if (codepoint <= 0xFFFF) {
160 // Single UTF16 character
161 *u16cur++ = (char16_t)codepoint;
162 } else {
163 // Multiple UTF16 characters with surrogates
164 codepoint = codepoint - 0x10000;
165 *u16cur++ = (char16_t)((codepoint >> offset) + 0xD800);
166 if (u16cur >= u16end) {
167 // Ooops... not enough room for this surrogate pair.
168 return u16cur - 1;
169 }
170 *u16cur++ = (char16_t)((codepoint & 0x3FF) + 0xDC00);
171 }
172
173 u8cur += len;
174 }
175 return u16cur;
176 }
177
Utf8CodePointLen(uint8_t ch)178 size_t QuickJSNativeString::Utf8CodePointLen(uint8_t ch)
179 {
180 constexpr uint8_t offset = 3;
181 return ((0xe5000000 >> ((ch >> offset) & 0x1e)) & offset) + 1;
182 }
183
Utf8ToUtf32CodePoint(const char * src,size_t length)184 uint32_t QuickJSNativeString::Utf8ToUtf32CodePoint(const char* src, size_t length)
185 {
186 uint32_t unicode = 0;
187 constexpr size_t lengthSizeOne = 1;
188 constexpr size_t lengthSizeTwo = 2;
189 constexpr size_t lengthSizeThree = 3;
190 constexpr size_t lengthSizeFour = 4;
191 constexpr size_t offsetZero = 0;
192 constexpr size_t offsetOne = 1;
193 constexpr size_t offsetTwo = 2;
194 constexpr size_t offsetThree = 3;
195 switch (length) {
196 case lengthSizeOne:
197 return src[offsetZero];
198 case lengthSizeTwo:
199 unicode = src[offsetZero] & 0x1f;
200 Utf8ShiftAndMask(&unicode, src[offsetOne]);
201 return unicode;
202 case lengthSizeThree:
203 unicode = src[offsetZero] & 0x0f;
204 Utf8ShiftAndMask(&unicode, src[offsetOne]);
205 Utf8ShiftAndMask(&unicode, src[offsetTwo]);
206 return unicode;
207 case lengthSizeFour:
208 unicode = src[offsetZero] & 0x07;
209 Utf8ShiftAndMask(&unicode, src[offsetOne]);
210 Utf8ShiftAndMask(&unicode, src[offsetTwo]);
211 Utf8ShiftAndMask(&unicode, src[offsetThree]);
212 return unicode;
213 default:
214 return 0xffff;
215 }
216 }
217
Utf8ShiftAndMask(uint32_t * codePoint,const uint8_t byte)218 void QuickJSNativeString::Utf8ShiftAndMask(uint32_t* codePoint, const uint8_t byte)
219 {
220 *codePoint <<= 6;
221 *codePoint |= 0x3F & byte;
222 }
223
Utf8ToUtf16Length(const char * str8,size_t str8Len)224 int QuickJSNativeString::Utf8ToUtf16Length(const char* str8, size_t str8Len)
225 {
226 const char* str8end = str8 + str8Len;
227 int utf16len = 0;
228 while (str8 < str8end) {
229 utf16len++;
230 size_t u8charlen = Utf8CodePointLen(*str8);
231 if (str8 + u8charlen - 1 >= str8end) {
232 HILOG_ERROR("Get str16 length failed because str8 unicode is illegal!");
233 return -1;
234 }
235 uint32_t codepoint = Utf8ToUtf32CodePoint(str8, u8charlen);
236 if (codepoint > 0xFFFF) {
237 utf16len++; // this will be a surrogate pair in utf16
238 }
239 str8 += u8charlen;
240 }
241 if (str8 != str8end) {
242 HILOG_ERROR("Get str16 length failed because str8length is illegal!");
243 return -1;
244 }
245 return utf16len;
246 }
247