| /* |
| * Copyright (C) 2008 The Android Open Source Project |
| * |
| * 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 "java_lang_StringFactory.h" |
| |
| #include "common_throws.h" |
| #include "jni_internal.h" |
| #include "mirror/object-inl.h" |
| #include "mirror/string.h" |
| #include "scoped_fast_native_object_access-inl.h" |
| #include "scoped_thread_state_change-inl.h" |
| #include "ScopedLocalRef.h" |
| #include "ScopedPrimitiveArray.h" |
| |
| namespace art { |
| |
| static jstring StringFactory_newStringFromBytes(JNIEnv* env, jclass, jbyteArray java_data, |
| jint high, jint offset, jint byte_count) { |
| ScopedFastNativeObjectAccess soa(env); |
| if (UNLIKELY(java_data == nullptr)) { |
| ThrowNullPointerException("data == null"); |
| return nullptr; |
| } |
| StackHandleScope<1> hs(soa.Self()); |
| Handle<mirror::ByteArray> byte_array(hs.NewHandle(soa.Decode<mirror::ByteArray>(java_data))); |
| int32_t data_size = byte_array->GetLength(); |
| if ((offset | byte_count) < 0 || byte_count > data_size - offset) { |
| soa.Self()->ThrowNewExceptionF("Ljava/lang/StringIndexOutOfBoundsException;", |
| "length=%d; regionStart=%d; regionLength=%d", data_size, |
| offset, byte_count); |
| return nullptr; |
| } |
| gc::AllocatorType allocator_type = Runtime::Current()->GetHeap()->GetCurrentAllocator(); |
| ObjPtr<mirror::String> result = mirror::String::AllocFromByteArray<true>(soa.Self(), |
| byte_count, |
| byte_array, |
| offset, |
| high, |
| allocator_type); |
| return soa.AddLocalReference<jstring>(result); |
| } |
| |
| // The char array passed as `java_data` must not be a null reference. |
| static jstring StringFactory_newStringFromChars(JNIEnv* env, jclass, jint offset, |
| jint char_count, jcharArray java_data) { |
| DCHECK(java_data != nullptr); |
| ScopedFastNativeObjectAccess soa(env); |
| StackHandleScope<1> hs(soa.Self()); |
| Handle<mirror::CharArray> char_array(hs.NewHandle(soa.Decode<mirror::CharArray>(java_data))); |
| gc::AllocatorType allocator_type = Runtime::Current()->GetHeap()->GetCurrentAllocator(); |
| ObjPtr<mirror::String> result = mirror::String::AllocFromCharArray<true>(soa.Self(), |
| char_count, |
| char_array, |
| offset, |
| allocator_type); |
| return soa.AddLocalReference<jstring>(result); |
| } |
| |
| static jstring StringFactory_newStringFromString(JNIEnv* env, jclass, jstring to_copy) { |
| ScopedFastNativeObjectAccess soa(env); |
| if (UNLIKELY(to_copy == nullptr)) { |
| ThrowNullPointerException("toCopy == null"); |
| return nullptr; |
| } |
| StackHandleScope<1> hs(soa.Self()); |
| Handle<mirror::String> string(hs.NewHandle(soa.Decode<mirror::String>(to_copy))); |
| gc::AllocatorType allocator_type = Runtime::Current()->GetHeap()->GetCurrentAllocator(); |
| ObjPtr<mirror::String> result = mirror::String::AllocFromString<true>(soa.Self(), |
| string->GetLength(), |
| string, |
| 0, |
| allocator_type); |
| return soa.AddLocalReference<jstring>(result); |
| } |
| |
| static JNINativeMethod gMethods[] = { |
| NATIVE_METHOD(StringFactory, newStringFromBytes, "!([BIII)Ljava/lang/String;"), |
| NATIVE_METHOD(StringFactory, newStringFromChars, "!(II[C)Ljava/lang/String;"), |
| NATIVE_METHOD(StringFactory, newStringFromString, "!(Ljava/lang/String;)Ljava/lang/String;"), |
| }; |
| |
| void register_java_lang_StringFactory(JNIEnv* env) { |
| REGISTER_NATIVE_METHODS("java/lang/StringFactory"); |
| } |
| |
| } // namespace art |