| // 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, NULL); |
| if (str == NULL) { |
| return NULL; |
| } |
| |
| 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 |