• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2008 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "java_lang_System.h"
18 
19 #include "nativehelper/jni_macros.h"
20 
21 #include "common_throws.h"
22 #include "gc/accounting/card_table-inl.h"
23 #include "jni_internal.h"
24 #include "mirror/array.h"
25 #include "mirror/class-inl.h"
26 #include "mirror/class.h"
27 #include "mirror/object-inl.h"
28 #include "mirror/object_array-inl.h"
29 #include "native_util.h"
30 #include "scoped_fast_native_object_access-inl.h"
31 
32 namespace art {
33 
34 /*
35  * We make guarantees about the atomicity of accesses to primitive variables.  These guarantees
36  * also apply to elements of arrays. In particular, 8-bit, 16-bit, and 32-bit accesses must not
37  * cause "word tearing".  Accesses to 64-bit array elements may be two 32-bit operations.
38  * References are never torn regardless of the number of bits used to represent them.
39  */
40 
ThrowArrayStoreException_NotAnArray(const char * identifier,ObjPtr<mirror::Object> array)41 static void ThrowArrayStoreException_NotAnArray(const char* identifier,
42                                                 ObjPtr<mirror::Object> array)
43     REQUIRES_SHARED(Locks::mutator_lock_) {
44   std::string actualType(mirror::Object::PrettyTypeOf(array));
45   Thread* self = Thread::Current();
46   self->ThrowNewExceptionF("Ljava/lang/ArrayStoreException;",
47                            "%s of type %s is not an array", identifier, actualType.c_str());
48 }
49 
System_arraycopy(JNIEnv * env,jclass,jobject javaSrc,jint srcPos,jobject javaDst,jint dstPos,jint length)50 static void System_arraycopy(JNIEnv* env, jclass, jobject javaSrc, jint srcPos, jobject javaDst,
51                              jint dstPos, jint length) {
52   // The API is defined in terms of length, but length is somewhat overloaded so we use count.
53   const jint count = length;
54   ScopedFastNativeObjectAccess soa(env);
55 
56   // Null pointer checks.
57   if (UNLIKELY(javaSrc == nullptr)) {
58     ThrowNullPointerException("src == null");
59     return;
60   }
61   if (UNLIKELY(javaDst == nullptr)) {
62     ThrowNullPointerException("dst == null");
63     return;
64   }
65 
66   // Make sure source and destination are both arrays.
67   ObjPtr<mirror::Object> srcObject = soa.Decode<mirror::Object>(javaSrc);
68   if (UNLIKELY(!srcObject->IsArrayInstance())) {
69     ThrowArrayStoreException_NotAnArray("source", srcObject);
70     return;
71   }
72   ObjPtr<mirror::Object> dstObject = soa.Decode<mirror::Object>(javaDst);
73   if (UNLIKELY(!dstObject->IsArrayInstance())) {
74     ThrowArrayStoreException_NotAnArray("destination", dstObject);
75     return;
76   }
77   ObjPtr<mirror::Array> srcArray = srcObject->AsArray();
78   ObjPtr<mirror::Array> dstArray = dstObject->AsArray();
79 
80   // Bounds checking.
81   if (UNLIKELY(srcPos < 0) || UNLIKELY(dstPos < 0) || UNLIKELY(count < 0) ||
82       UNLIKELY(srcPos > srcArray->GetLength() - count) ||
83       UNLIKELY(dstPos > dstArray->GetLength() - count)) {
84     soa.Self()->ThrowNewExceptionF("Ljava/lang/ArrayIndexOutOfBoundsException;",
85                                    "src.length=%d srcPos=%d dst.length=%d dstPos=%d length=%d",
86                                    srcArray->GetLength(), srcPos, dstArray->GetLength(), dstPos,
87                                    count);
88     return;
89   }
90 
91   ObjPtr<mirror::Class> dstComponentType = dstArray->GetClass()->GetComponentType();
92   ObjPtr<mirror::Class> srcComponentType = srcArray->GetClass()->GetComponentType();
93   Primitive::Type dstComponentPrimitiveType = dstComponentType->GetPrimitiveType();
94 
95   if (LIKELY(srcComponentType == dstComponentType)) {
96     // Trivial assignability.
97     switch (dstComponentPrimitiveType) {
98       case Primitive::kPrimVoid:
99         LOG(FATAL) << "Unreachable, cannot have arrays of type void";
100         UNREACHABLE();
101       case Primitive::kPrimBoolean:
102       case Primitive::kPrimByte:
103         DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 1U);
104         dstArray->AsByteSizedArray()->Memmove(dstPos, srcArray->AsByteSizedArray(), srcPos, count);
105         return;
106       case Primitive::kPrimChar:
107       case Primitive::kPrimShort:
108         DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 2U);
109         dstArray->AsShortSizedArray()->Memmove(dstPos, srcArray->AsShortSizedArray(), srcPos, count);
110         return;
111       case Primitive::kPrimInt:
112         DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 4U);
113         dstArray->AsIntArray()->Memmove(dstPos, srcArray->AsIntArray(), srcPos, count);
114         return;
115       case Primitive::kPrimFloat:
116         DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 4U);
117         dstArray->AsFloatArray()->Memmove(dstPos, srcArray->AsFloatArray(), srcPos, count);
118         return;
119       case Primitive::kPrimLong:
120         DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 8U);
121         dstArray->AsLongArray()->Memmove(dstPos, srcArray->AsLongArray(), srcPos, count);
122         return;
123       case Primitive::kPrimDouble:
124         DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 8U);
125         dstArray->AsDoubleArray()->Memmove(dstPos, srcArray->AsDoubleArray(), srcPos, count);
126         return;
127       case Primitive::kPrimNot: {
128         mirror::ObjectArray<mirror::Object>* dstObjArray = dstArray->AsObjectArray<mirror::Object>();
129         mirror::ObjectArray<mirror::Object>* srcObjArray = srcArray->AsObjectArray<mirror::Object>();
130         dstObjArray->AssignableMemmove(dstPos, srcObjArray, srcPos, count);
131         return;
132       }
133       default:
134         LOG(FATAL) << "Unknown array type: " << srcArray->PrettyTypeOf();
135         UNREACHABLE();
136     }
137   }
138   // If one of the arrays holds a primitive type the other array must hold the exact same type.
139   if (UNLIKELY((dstComponentPrimitiveType != Primitive::kPrimNot) ||
140                srcComponentType->IsPrimitive())) {
141     std::string srcType(srcArray->PrettyTypeOf());
142     std::string dstType(dstArray->PrettyTypeOf());
143     soa.Self()->ThrowNewExceptionF("Ljava/lang/ArrayStoreException;",
144                                    "Incompatible types: src=%s, dst=%s",
145                                    srcType.c_str(), dstType.c_str());
146     return;
147   }
148   // Arrays hold distinct types and so therefore can't alias - use memcpy instead of memmove.
149   ObjPtr<mirror::ObjectArray<mirror::Object>> dstObjArray =
150       dstArray->AsObjectArray<mirror::Object>();
151   ObjPtr<mirror::ObjectArray<mirror::Object>> srcObjArray =
152       srcArray->AsObjectArray<mirror::Object>();
153   // If we're assigning into say Object[] then we don't need per element checks.
154   if (dstComponentType->IsAssignableFrom(srcComponentType)) {
155     dstObjArray->AssignableMemcpy(dstPos, srcObjArray, srcPos, count);
156     return;
157   }
158   // This code is never run under a transaction.
159   DCHECK(!Runtime::Current()->IsActiveTransaction());
160   dstObjArray->AssignableCheckingMemcpy<false>(dstPos, srcObjArray, srcPos, count, true);
161 }
162 
163 // Template to convert general array to that of its specific primitive type.
164 template <typename T>
AsPrimitiveArray(ObjPtr<mirror::Array> array)165 inline ObjPtr<T> AsPrimitiveArray(ObjPtr<mirror::Array> array)
166     REQUIRES_SHARED(Locks::mutator_lock_) {
167   return ObjPtr<T>::DownCast(array);
168 }
169 
170 template <typename T, Primitive::Type kPrimType>
System_arraycopyTUnchecked(JNIEnv * env,jobject javaSrc,jint srcPos,jobject javaDst,jint dstPos,jint count)171 inline void System_arraycopyTUnchecked(JNIEnv* env, jobject javaSrc, jint srcPos,
172                                        jobject javaDst, jint dstPos, jint count) {
173   ScopedFastNativeObjectAccess soa(env);
174   ObjPtr<mirror::Object> srcObject = soa.Decode<mirror::Object>(javaSrc);
175   ObjPtr<mirror::Object> dstObject = soa.Decode<mirror::Object>(javaDst);
176   DCHECK(dstObject != nullptr);
177   ObjPtr<mirror::Array> srcArray = srcObject->AsArray();
178   ObjPtr<mirror::Array> dstArray = dstObject->AsArray();
179   DCHECK_GE(count, 0);
180   DCHECK_EQ(srcArray->GetClass(), dstArray->GetClass());
181   DCHECK_EQ(srcArray->GetClass()->GetComponentType()->GetPrimitiveType(), kPrimType);
182   AsPrimitiveArray<T>(dstArray)->Memmove(dstPos, AsPrimitiveArray<T>(srcArray), srcPos, count);
183 }
184 
System_arraycopyCharUnchecked(JNIEnv * env,jclass,jcharArray javaSrc,jint srcPos,jcharArray javaDst,jint dstPos,jint count)185 static void System_arraycopyCharUnchecked(JNIEnv* env, jclass, jcharArray javaSrc, jint srcPos,
186                                           jcharArray javaDst, jint dstPos, jint count) {
187   System_arraycopyTUnchecked<mirror::CharArray, Primitive::kPrimChar>(env, javaSrc, srcPos,
188       javaDst, dstPos, count);
189 }
190 
System_arraycopyByteUnchecked(JNIEnv * env,jclass,jbyteArray javaSrc,jint srcPos,jbyteArray javaDst,jint dstPos,jint count)191 static void System_arraycopyByteUnchecked(JNIEnv* env, jclass, jbyteArray javaSrc, jint srcPos,
192                                           jbyteArray javaDst, jint dstPos, jint count) {
193   System_arraycopyTUnchecked<mirror::ByteArray, Primitive::kPrimByte>(env, javaSrc, srcPos,
194       javaDst, dstPos, count);
195 }
196 
System_arraycopyShortUnchecked(JNIEnv * env,jclass,jshortArray javaSrc,jint srcPos,jshortArray javaDst,jint dstPos,jint count)197 static void System_arraycopyShortUnchecked(JNIEnv* env, jclass, jshortArray javaSrc, jint srcPos,
198                                            jshortArray javaDst, jint dstPos, jint count) {
199   System_arraycopyTUnchecked<mirror::ShortArray, Primitive::kPrimShort>(env, javaSrc, srcPos,
200       javaDst, dstPos, count);
201 }
202 
System_arraycopyIntUnchecked(JNIEnv * env,jclass,jintArray javaSrc,jint srcPos,jintArray javaDst,jint dstPos,jint count)203 static void System_arraycopyIntUnchecked(JNIEnv* env, jclass, jintArray javaSrc, jint srcPos,
204                                          jintArray javaDst, jint dstPos, jint count) {
205   System_arraycopyTUnchecked<mirror::IntArray, Primitive::kPrimInt>(env, javaSrc, srcPos,
206       javaDst, dstPos, count);
207 }
208 
System_arraycopyLongUnchecked(JNIEnv * env,jclass,jlongArray javaSrc,jint srcPos,jlongArray javaDst,jint dstPos,jint count)209 static void System_arraycopyLongUnchecked(JNIEnv* env, jclass, jlongArray javaSrc, jint srcPos,
210                                           jlongArray javaDst, jint dstPos, jint count) {
211   System_arraycopyTUnchecked<mirror::LongArray, Primitive::kPrimLong>(env, javaSrc, srcPos,
212       javaDst, dstPos, count);
213 }
214 
System_arraycopyFloatUnchecked(JNIEnv * env,jclass,jfloatArray javaSrc,jint srcPos,jfloatArray javaDst,jint dstPos,jint count)215 static void System_arraycopyFloatUnchecked(JNIEnv* env, jclass, jfloatArray javaSrc, jint srcPos,
216                                            jfloatArray javaDst, jint dstPos, jint count) {
217   System_arraycopyTUnchecked<mirror::FloatArray, Primitive::kPrimFloat>(env, javaSrc, srcPos,
218       javaDst, dstPos, count);
219 }
220 
System_arraycopyDoubleUnchecked(JNIEnv * env,jclass,jdoubleArray javaSrc,jint srcPos,jdoubleArray javaDst,jint dstPos,jint count)221 static void System_arraycopyDoubleUnchecked(JNIEnv* env, jclass, jdoubleArray javaSrc, jint srcPos,
222                                             jdoubleArray javaDst, jint dstPos, jint count) {
223   System_arraycopyTUnchecked<mirror::DoubleArray, Primitive::kPrimDouble>(env, javaSrc, srcPos,
224       javaDst, dstPos, count);
225 }
226 
System_arraycopyBooleanUnchecked(JNIEnv * env,jclass,jbooleanArray javaSrc,jint srcPos,jbooleanArray javaDst,jint dstPos,jint count)227 static void System_arraycopyBooleanUnchecked(JNIEnv* env,
228                                              jclass,
229                                              jbooleanArray javaSrc,
230                                              jint srcPos,
231                                              jbooleanArray javaDst,
232                                              jint dstPos,
233                                              jint count) {
234   System_arraycopyTUnchecked<mirror::BooleanArray, Primitive::kPrimBoolean>(env, javaSrc, srcPos,
235       javaDst, dstPos, count);
236 }
237 
238 static JNINativeMethod gMethods[] = {
239   FAST_NATIVE_METHOD(System, arraycopy, "(Ljava/lang/Object;ILjava/lang/Object;II)V"),
240   FAST_NATIVE_METHOD(System, arraycopyCharUnchecked, "([CI[CII)V"),
241   FAST_NATIVE_METHOD(System, arraycopyByteUnchecked, "([BI[BII)V"),
242   FAST_NATIVE_METHOD(System, arraycopyShortUnchecked, "([SI[SII)V"),
243   FAST_NATIVE_METHOD(System, arraycopyIntUnchecked, "([II[III)V"),
244   FAST_NATIVE_METHOD(System, arraycopyLongUnchecked, "([JI[JII)V"),
245   FAST_NATIVE_METHOD(System, arraycopyFloatUnchecked, "([FI[FII)V"),
246   FAST_NATIVE_METHOD(System, arraycopyDoubleUnchecked, "([DI[DII)V"),
247   FAST_NATIVE_METHOD(System, arraycopyBooleanUnchecked, "([ZI[ZII)V"),
248 };
249 
register_java_lang_System(JNIEnv * env)250 void register_java_lang_System(JNIEnv* env) {
251   REGISTER_NATIVE_METHODS("java/lang/System");
252 }
253 
254 }  // namespace art
255