• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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