|  | // Copyright 2019 The Bazel Authors. All rights reserved. | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //    http://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | #include "src/main/native/latin1_jni_path.h" | 
|  |  | 
|  | #include <string.h> | 
|  |  | 
|  | namespace blaze_jni { | 
|  |  | 
|  | jstring NewStringLatin1(JNIEnv *env, const char *str) { | 
|  | int len = strlen(str); | 
|  | jchar buf[512]; | 
|  | jchar *str1; | 
|  |  | 
|  | if (len > 512) { | 
|  | str1 = new jchar[len]; | 
|  | } else { | 
|  | str1 = buf; | 
|  | } | 
|  |  | 
|  | for (int i = 0; i < len; i++) { | 
|  | str1[i] = (unsigned char)str[i]; | 
|  | } | 
|  | jstring result = env->NewString(str1, len); | 
|  | if (str1 != buf) { | 
|  | delete[] str1; | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | static jfieldID String_coder_field; | 
|  | static jfieldID String_value_field; | 
|  |  | 
|  | static bool CompactStringsEnabled(JNIEnv *env) { | 
|  | if (jclass klass = env->FindClass("java/lang/String")) { | 
|  | if (jfieldID csf = env->GetStaticFieldID(klass, "COMPACT_STRINGS", "Z")) { | 
|  | if (env->GetStaticBooleanField(klass, csf)) { | 
|  | if ((String_coder_field = env->GetFieldID(klass, "coder", "B"))) { | 
|  | if ((String_value_field = env->GetFieldID(klass, "value", "[B"))) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | env->ExceptionClear(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | char *GetStringLatin1Chars(JNIEnv *env, jstring jstr) { | 
|  | jint len = env->GetStringLength(jstr); | 
|  |  | 
|  | // Fast path for latin1 strings. | 
|  | static bool cs_enabled = CompactStringsEnabled(env); | 
|  | const int LATIN1 = 0; | 
|  | if (cs_enabled && env->GetByteField(jstr, String_coder_field) == LATIN1) { | 
|  | char *result = new char[len + 1]; | 
|  | if (jobject jvalue = env->GetObjectField(jstr, String_value_field)) { | 
|  | env->GetByteArrayRegion((jbyteArray)jvalue, 0, len, (jbyte *)result); | 
|  | } | 
|  | result[len] = 0; | 
|  | return result; | 
|  | } | 
|  |  | 
|  | const jchar *str = env->GetStringCritical(jstr, nullptr); | 
|  | if (str == nullptr) { | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | char *result = new char[len + 1]; | 
|  | for (int i = 0; i < len; i++) { | 
|  | jchar unicode = str[i];  // (unsigned) | 
|  | result[i] = unicode <= 0x00ff ? unicode : '?'; | 
|  | } | 
|  |  | 
|  | result[len] = 0; | 
|  | env->ReleaseStringCritical(jstr, str); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Release the Latin1 chars returned by a prior call to | 
|  | * GetStringLatin1Chars. | 
|  | */ | 
|  | void ReleaseStringLatin1Chars(const char *s) { delete[] s; } | 
|  |  | 
|  | }  // namespace blaze_jni |