xref: /aosp_15_r20/external/cronet/base/android/jni_string.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "base/android/jni_string.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <string_view>
8*6777b538SAndroid Build Coastguard Worker 
9*6777b538SAndroid Build Coastguard Worker #include "base/android/jni_android.h"
10*6777b538SAndroid Build Coastguard Worker #include "base/logging.h"
11*6777b538SAndroid Build Coastguard Worker #include "base/numerics/safe_conversions.h"
12*6777b538SAndroid Build Coastguard Worker #include "base/strings/utf_string_conversions.h"
13*6777b538SAndroid Build Coastguard Worker 
14*6777b538SAndroid Build Coastguard Worker // Size of buffer to allocate on the stack for string conversion.
15*6777b538SAndroid Build Coastguard Worker #define BUFFER_SIZE 1024
16*6777b538SAndroid Build Coastguard Worker 
17*6777b538SAndroid Build Coastguard Worker namespace {
18*6777b538SAndroid Build Coastguard Worker 
19*6777b538SAndroid Build Coastguard Worker // Internal version that does not use a scoped local pointer.
ConvertUTF16ToJavaStringImpl(JNIEnv * env,std::u16string_view str)20*6777b538SAndroid Build Coastguard Worker jstring ConvertUTF16ToJavaStringImpl(JNIEnv* env, std::u16string_view str) {
21*6777b538SAndroid Build Coastguard Worker   jstring result = env->NewString(reinterpret_cast<const jchar*>(str.data()),
22*6777b538SAndroid Build Coastguard Worker                                   base::checked_cast<jsize>(str.length()));
23*6777b538SAndroid Build Coastguard Worker   base::android::CheckException(env);
24*6777b538SAndroid Build Coastguard Worker   return result;
25*6777b538SAndroid Build Coastguard Worker }
26*6777b538SAndroid Build Coastguard Worker 
27*6777b538SAndroid Build Coastguard Worker }  // namespace
28*6777b538SAndroid Build Coastguard Worker 
29*6777b538SAndroid Build Coastguard Worker namespace base {
30*6777b538SAndroid Build Coastguard Worker namespace android {
31*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF8(JNIEnv * env,jstring str,std::string * result)32*6777b538SAndroid Build Coastguard Worker void ConvertJavaStringToUTF8(JNIEnv* env, jstring str, std::string* result) {
33*6777b538SAndroid Build Coastguard Worker   DCHECK(str);
34*6777b538SAndroid Build Coastguard Worker   if (!str) {
35*6777b538SAndroid Build Coastguard Worker     LOG(WARNING) << "ConvertJavaStringToUTF8 called with null string.";
36*6777b538SAndroid Build Coastguard Worker     result->clear();
37*6777b538SAndroid Build Coastguard Worker     return;
38*6777b538SAndroid Build Coastguard Worker   }
39*6777b538SAndroid Build Coastguard Worker   const jsize length = env->GetStringLength(str);
40*6777b538SAndroid Build Coastguard Worker   if (length <= 0) {
41*6777b538SAndroid Build Coastguard Worker     result->clear();
42*6777b538SAndroid Build Coastguard Worker     CheckException(env);
43*6777b538SAndroid Build Coastguard Worker     return;
44*6777b538SAndroid Build Coastguard Worker   }
45*6777b538SAndroid Build Coastguard Worker   // JNI's GetStringUTFChars() and GetStringUTFRegion returns strings in Java
46*6777b538SAndroid Build Coastguard Worker   // "modified" UTF8, so instead get the String in UTF16 and convert using
47*6777b538SAndroid Build Coastguard Worker   // chromium's conversion function that yields plain (non Java-modified) UTF8.
48*6777b538SAndroid Build Coastguard Worker   if (length <= BUFFER_SIZE) {
49*6777b538SAndroid Build Coastguard Worker     // fast path, allocate temporary buffer on the stack and use GetStringRegion
50*6777b538SAndroid Build Coastguard Worker     // to copy the utf-16 characters into it with no heap allocation.
51*6777b538SAndroid Build Coastguard Worker     // https://developer.android.com/training/articles/perf-jni#utf-8-and-utf-16-strings:~:text=stack%2Dallocated%20buffer
52*6777b538SAndroid Build Coastguard Worker     std::array<jchar, BUFFER_SIZE> chars;
53*6777b538SAndroid Build Coastguard Worker     // GetStringRegion does not copy a null terminated string so the length must
54*6777b538SAndroid Build Coastguard Worker     // be explicitly passed to UTF16ToUTF8.
55*6777b538SAndroid Build Coastguard Worker     env->GetStringRegion(str, 0, length, chars.data());
56*6777b538SAndroid Build Coastguard Worker     UTF16ToUTF8(reinterpret_cast<const char16_t*>(chars.data()),
57*6777b538SAndroid Build Coastguard Worker                 static_cast<size_t>(length), result);
58*6777b538SAndroid Build Coastguard Worker   } else {
59*6777b538SAndroid Build Coastguard Worker     // slow path
60*6777b538SAndroid Build Coastguard Worker     // GetStringChars doesn't NULL-terminate the strings it returns, so the
61*6777b538SAndroid Build Coastguard Worker     // length must be explicitly passed to UTF16ToUTF8.
62*6777b538SAndroid Build Coastguard Worker     const jchar* chars = env->GetStringChars(str, NULL);
63*6777b538SAndroid Build Coastguard Worker     DCHECK(chars);
64*6777b538SAndroid Build Coastguard Worker     UTF16ToUTF8(reinterpret_cast<const char16_t*>(chars),
65*6777b538SAndroid Build Coastguard Worker                 static_cast<size_t>(length), result);
66*6777b538SAndroid Build Coastguard Worker     env->ReleaseStringChars(str, chars);
67*6777b538SAndroid Build Coastguard Worker   }
68*6777b538SAndroid Build Coastguard Worker   CheckException(env);
69*6777b538SAndroid Build Coastguard Worker }
70*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF8(JNIEnv * env,jstring str)71*6777b538SAndroid Build Coastguard Worker std::string ConvertJavaStringToUTF8(JNIEnv* env, jstring str) {
72*6777b538SAndroid Build Coastguard Worker   std::string result;
73*6777b538SAndroid Build Coastguard Worker   ConvertJavaStringToUTF8(env, str, &result);
74*6777b538SAndroid Build Coastguard Worker   return result;
75*6777b538SAndroid Build Coastguard Worker }
76*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF8(const JavaRef<jstring> & str)77*6777b538SAndroid Build Coastguard Worker std::string ConvertJavaStringToUTF8(const JavaRef<jstring>& str) {
78*6777b538SAndroid Build Coastguard Worker   return ConvertJavaStringToUTF8(AttachCurrentThread(), str.obj());
79*6777b538SAndroid Build Coastguard Worker }
80*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF8(JNIEnv * env,const JavaRef<jstring> & str)81*6777b538SAndroid Build Coastguard Worker std::string ConvertJavaStringToUTF8(JNIEnv* env, const JavaRef<jstring>& str) {
82*6777b538SAndroid Build Coastguard Worker   return ConvertJavaStringToUTF8(env, str.obj());
83*6777b538SAndroid Build Coastguard Worker }
84*6777b538SAndroid Build Coastguard Worker 
ConvertUTF8ToJavaString(JNIEnv * env,std::string_view str)85*6777b538SAndroid Build Coastguard Worker ScopedJavaLocalRef<jstring> ConvertUTF8ToJavaString(JNIEnv* env,
86*6777b538SAndroid Build Coastguard Worker                                                     std::string_view str) {
87*6777b538SAndroid Build Coastguard Worker   // JNI's NewStringUTF expects "modified" UTF8 so instead create the string
88*6777b538SAndroid Build Coastguard Worker   // via our own UTF16 conversion utility.
89*6777b538SAndroid Build Coastguard Worker   // Further, Dalvik requires the string passed into NewStringUTF() to come from
90*6777b538SAndroid Build Coastguard Worker   // a trusted source. We can't guarantee that all UTF8 will be sanitized before
91*6777b538SAndroid Build Coastguard Worker   // it gets here, so constructing via UTF16 side-steps this issue.
92*6777b538SAndroid Build Coastguard Worker   // (Dalvik stores strings internally as UTF16 anyway, so there shouldn't be
93*6777b538SAndroid Build Coastguard Worker   // a significant performance hit by doing it this way).
94*6777b538SAndroid Build Coastguard Worker   return ScopedJavaLocalRef<jstring>(env, ConvertUTF16ToJavaStringImpl(
95*6777b538SAndroid Build Coastguard Worker       env, UTF8ToUTF16(str)));
96*6777b538SAndroid Build Coastguard Worker }
97*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF16(JNIEnv * env,jstring str,std::u16string * result)98*6777b538SAndroid Build Coastguard Worker void ConvertJavaStringToUTF16(JNIEnv* env,
99*6777b538SAndroid Build Coastguard Worker                               jstring str,
100*6777b538SAndroid Build Coastguard Worker                               std::u16string* result) {
101*6777b538SAndroid Build Coastguard Worker   DCHECK(str);
102*6777b538SAndroid Build Coastguard Worker   if (!str) {
103*6777b538SAndroid Build Coastguard Worker     LOG(WARNING) << "ConvertJavaStringToUTF16 called with null string.";
104*6777b538SAndroid Build Coastguard Worker     result->clear();
105*6777b538SAndroid Build Coastguard Worker     return;
106*6777b538SAndroid Build Coastguard Worker   }
107*6777b538SAndroid Build Coastguard Worker   const jsize length = env->GetStringLength(str);
108*6777b538SAndroid Build Coastguard Worker   if (length <= 0) {
109*6777b538SAndroid Build Coastguard Worker     result->clear();
110*6777b538SAndroid Build Coastguard Worker     CheckException(env);
111*6777b538SAndroid Build Coastguard Worker     return;
112*6777b538SAndroid Build Coastguard Worker   }
113*6777b538SAndroid Build Coastguard Worker   if (length <= BUFFER_SIZE) {
114*6777b538SAndroid Build Coastguard Worker     // fast path, allocate temporary buffer on the stack and use GetStringRegion
115*6777b538SAndroid Build Coastguard Worker     // to copy the utf-16 characters into it with no heap allocation.
116*6777b538SAndroid Build Coastguard Worker     // https://developer.android.com/training/articles/perf-jni#utf-8-and-utf-16-strings:~:text=stack%2Dallocated%20buffer
117*6777b538SAndroid Build Coastguard Worker     std::array<jchar, BUFFER_SIZE> chars;
118*6777b538SAndroid Build Coastguard Worker     env->GetStringRegion(str, 0, length, chars.data());
119*6777b538SAndroid Build Coastguard Worker     // GetStringRegion does not copy a null terminated string so the length must
120*6777b538SAndroid Build Coastguard Worker     // be explicitly passed to assign.
121*6777b538SAndroid Build Coastguard Worker     result->assign(reinterpret_cast<const char16_t*>(chars.data()),
122*6777b538SAndroid Build Coastguard Worker                    static_cast<size_t>(length));
123*6777b538SAndroid Build Coastguard Worker   } else {
124*6777b538SAndroid Build Coastguard Worker     // slow path
125*6777b538SAndroid Build Coastguard Worker     const jchar* chars = env->GetStringChars(str, NULL);
126*6777b538SAndroid Build Coastguard Worker     DCHECK(chars);
127*6777b538SAndroid Build Coastguard Worker     // GetStringChars doesn't NULL-terminate the strings it returns, so the
128*6777b538SAndroid Build Coastguard Worker     // length must be explicitly passed to assign.
129*6777b538SAndroid Build Coastguard Worker     result->assign(reinterpret_cast<const char16_t*>(chars),
130*6777b538SAndroid Build Coastguard Worker                    static_cast<size_t>(length));
131*6777b538SAndroid Build Coastguard Worker     env->ReleaseStringChars(str, chars);
132*6777b538SAndroid Build Coastguard Worker   }
133*6777b538SAndroid Build Coastguard Worker   CheckException(env);
134*6777b538SAndroid Build Coastguard Worker }
135*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF16(JNIEnv * env,jstring str)136*6777b538SAndroid Build Coastguard Worker std::u16string ConvertJavaStringToUTF16(JNIEnv* env, jstring str) {
137*6777b538SAndroid Build Coastguard Worker   std::u16string result;
138*6777b538SAndroid Build Coastguard Worker   ConvertJavaStringToUTF16(env, str, &result);
139*6777b538SAndroid Build Coastguard Worker   return result;
140*6777b538SAndroid Build Coastguard Worker }
141*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF16(const JavaRef<jstring> & str)142*6777b538SAndroid Build Coastguard Worker std::u16string ConvertJavaStringToUTF16(const JavaRef<jstring>& str) {
143*6777b538SAndroid Build Coastguard Worker   return ConvertJavaStringToUTF16(AttachCurrentThread(), str.obj());
144*6777b538SAndroid Build Coastguard Worker }
145*6777b538SAndroid Build Coastguard Worker 
ConvertJavaStringToUTF16(JNIEnv * env,const JavaRef<jstring> & str)146*6777b538SAndroid Build Coastguard Worker std::u16string ConvertJavaStringToUTF16(JNIEnv* env,
147*6777b538SAndroid Build Coastguard Worker                                         const JavaRef<jstring>& str) {
148*6777b538SAndroid Build Coastguard Worker   return ConvertJavaStringToUTF16(env, str.obj());
149*6777b538SAndroid Build Coastguard Worker }
150*6777b538SAndroid Build Coastguard Worker 
ConvertUTF16ToJavaString(JNIEnv * env,std::u16string_view str)151*6777b538SAndroid Build Coastguard Worker ScopedJavaLocalRef<jstring> ConvertUTF16ToJavaString(JNIEnv* env,
152*6777b538SAndroid Build Coastguard Worker                                                      std::u16string_view str) {
153*6777b538SAndroid Build Coastguard Worker   return ScopedJavaLocalRef<jstring>(env,
154*6777b538SAndroid Build Coastguard Worker                                      ConvertUTF16ToJavaStringImpl(env, str));
155*6777b538SAndroid Build Coastguard Worker }
156*6777b538SAndroid Build Coastguard Worker 
157*6777b538SAndroid Build Coastguard Worker }  // namespace android
158*6777b538SAndroid Build Coastguard Worker }  // namespace base
159