blob: 264b42746011a2a4e08f8cffbf6e38aa9f9af239 [file] [log] [blame]
Elliott Hughesbf86d042011-08-31 17:53:14 -07001/*
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
Andreas Gampe277ccbd2014-11-03 21:36:10 -080017#include "java_lang_System.h"
18
Andreas Gampea14100c2017-04-24 15:09:56 -070019#include "nativehelper/jni_macros.h"
20
Ian Rogers62d6c772013-02-27 08:32:07 -080021#include "common_throws.h"
Ian Rogers1d54e732013-05-02 21:10:01 -070022#include "gc/accounting/card_table-inl.h"
Elliott Hughesbf86d042011-08-31 17:53:14 -070023#include "jni_internal.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080024#include "mirror/array.h"
25#include "mirror/class.h"
26#include "mirror/class-inl.h"
27#include "mirror/object-inl.h"
Ian Rogers693ff612013-02-01 10:56:12 -080028#include "mirror/object_array-inl.h"
Mathieu Chartier0795f232016-09-27 18:43:30 -070029#include "scoped_fast_native_object_access-inl.h"
Elliott Hughesbf86d042011-08-31 17:53:14 -070030
Elliott Hughesbf86d042011-08-31 17:53:14 -070031namespace art {
32
Ian Rogersef7d42f2014-01-06 12:55:46 -080033/*
34 * We make guarantees about the atomicity of accesses to primitive variables. These guarantees
35 * also apply to elements of arrays. In particular, 8-bit, 16-bit, and 32-bit accesses must not
36 * cause "word tearing". Accesses to 64-bit array elements may be two 32-bit operations.
37 * References are never torn regardless of the number of bits used to represent them.
38 */
39
Mathieu Chartier6b3d12b2016-10-13 13:59:58 -070040static void ThrowArrayStoreException_NotAnArray(const char* identifier,
41 ObjPtr<mirror::Object> array)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070042 REQUIRES_SHARED(Locks::mutator_lock_) {
David Sehr709b0702016-10-13 09:12:37 -070043 std::string actualType(mirror::Object::PrettyTypeOf(array));
Ian Rogers62d6c772013-02-27 08:32:07 -080044 Thread* self = Thread::Current();
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +000045 self->ThrowNewExceptionF("Ljava/lang/ArrayStoreException;",
Ian Rogers62d6c772013-02-27 08:32:07 -080046 "%s of type %s is not an array", identifier, actualType.c_str());
Elliott Hughesbf86d042011-08-31 17:53:14 -070047}
48
Ian Rogersef7d42f2014-01-06 12:55:46 -080049static void System_arraycopy(JNIEnv* env, jclass, jobject javaSrc, jint srcPos, jobject javaDst,
50 jint dstPos, jint length) {
51 // The API is defined in terms of length, but length is somewhat overloaded so we use count.
52 const jint count = length;
Ian Rogers1eb512d2013-10-18 15:42:20 -070053 ScopedFastNativeObjectAccess soa(env);
Elliott Hughesbf86d042011-08-31 17:53:14 -070054
55 // Null pointer checks.
Ian Rogersef7d42f2014-01-06 12:55:46 -080056 if (UNLIKELY(javaSrc == nullptr)) {
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +000057 ThrowNullPointerException("src == null");
Elliott Hughesbf86d042011-08-31 17:53:14 -070058 return;
59 }
Ian Rogersef7d42f2014-01-06 12:55:46 -080060 if (UNLIKELY(javaDst == nullptr)) {
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +000061 ThrowNullPointerException("dst == null");
Elliott Hughesbf86d042011-08-31 17:53:14 -070062 return;
63 }
64
65 // Make sure source and destination are both arrays.
Mathieu Chartier0795f232016-09-27 18:43:30 -070066 ObjPtr<mirror::Object> srcObject = soa.Decode<mirror::Object>(javaSrc);
Ian Rogers62d6c772013-02-27 08:32:07 -080067 if (UNLIKELY(!srcObject->IsArrayInstance())) {
Mathieu Chartier6b3d12b2016-10-13 13:59:58 -070068 ThrowArrayStoreException_NotAnArray("source", srcObject);
Elliott Hughesbf86d042011-08-31 17:53:14 -070069 return;
70 }
Mathieu Chartier0795f232016-09-27 18:43:30 -070071 ObjPtr<mirror::Object> dstObject = soa.Decode<mirror::Object>(javaDst);
Ian Rogers62d6c772013-02-27 08:32:07 -080072 if (UNLIKELY(!dstObject->IsArrayInstance())) {
Mathieu Chartier6b3d12b2016-10-13 13:59:58 -070073 ThrowArrayStoreException_NotAnArray("destination", dstObject);
Elliott Hughesbf86d042011-08-31 17:53:14 -070074 return;
75 }
Mathieu Chartierbc5a7952016-10-17 15:46:31 -070076 ObjPtr<mirror::Array> srcArray = srcObject->AsArray();
77 ObjPtr<mirror::Array> dstArray = dstObject->AsArray();
Elliott Hughesbf86d042011-08-31 17:53:14 -070078
79 // Bounds checking.
Ian Rogersef7d42f2014-01-06 12:55:46 -080080 if (UNLIKELY(srcPos < 0) || UNLIKELY(dstPos < 0) || UNLIKELY(count < 0) ||
81 UNLIKELY(srcPos > srcArray->GetLength() - count) ||
82 UNLIKELY(dstPos > dstArray->GetLength() - count)) {
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +000083 soa.Self()->ThrowNewExceptionF("Ljava/lang/ArrayIndexOutOfBoundsException;",
Ian Rogers62d6c772013-02-27 08:32:07 -080084 "src.length=%d srcPos=%d dst.length=%d dstPos=%d length=%d",
Ian Rogersef7d42f2014-01-06 12:55:46 -080085 srcArray->GetLength(), srcPos, dstArray->GetLength(), dstPos,
86 count);
Elliott Hughesbf86d042011-08-31 17:53:14 -070087 return;
88 }
89
Mathieu Chartierbc5a7952016-10-17 15:46:31 -070090 ObjPtr<mirror::Class> dstComponentType = dstArray->GetClass()->GetComponentType();
91 ObjPtr<mirror::Class> srcComponentType = srcArray->GetClass()->GetComponentType();
Ian Rogersef7d42f2014-01-06 12:55:46 -080092 Primitive::Type dstComponentPrimitiveType = dstComponentType->GetPrimitiveType();
Elliott Hughesbf86d042011-08-31 17:53:14 -070093
Ian Rogersef7d42f2014-01-06 12:55:46 -080094 if (LIKELY(srcComponentType == dstComponentType)) {
95 // Trivial assignability.
96 switch (dstComponentPrimitiveType) {
97 case Primitive::kPrimVoid:
98 LOG(FATAL) << "Unreachable, cannot have arrays of type void";
Ian Rogers2c4257b2014-10-24 14:20:06 -070099 UNREACHABLE();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800100 case Primitive::kPrimBoolean:
101 case Primitive::kPrimByte:
102 DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 1U);
103 dstArray->AsByteSizedArray()->Memmove(dstPos, srcArray->AsByteSizedArray(), srcPos, count);
104 return;
105 case Primitive::kPrimChar:
106 case Primitive::kPrimShort:
107 DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 2U);
108 dstArray->AsShortSizedArray()->Memmove(dstPos, srcArray->AsShortSizedArray(), srcPos, count);
109 return;
110 case Primitive::kPrimInt:
Ian Rogersef7d42f2014-01-06 12:55:46 -0800111 DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 4U);
112 dstArray->AsIntArray()->Memmove(dstPos, srcArray->AsIntArray(), srcPos, count);
113 return;
Andreas Gampec952ac92015-07-16 17:41:25 -0700114 case Primitive::kPrimFloat:
115 DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 4U);
116 dstArray->AsFloatArray()->Memmove(dstPos, srcArray->AsFloatArray(), srcPos, count);
117 return;
Ian Rogersef7d42f2014-01-06 12:55:46 -0800118 case Primitive::kPrimLong:
Ian Rogersef7d42f2014-01-06 12:55:46 -0800119 DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 8U);
120 dstArray->AsLongArray()->Memmove(dstPos, srcArray->AsLongArray(), srcPos, count);
121 return;
Andreas Gampec952ac92015-07-16 17:41:25 -0700122 case Primitive::kPrimDouble:
123 DCHECK_EQ(Primitive::ComponentSize(dstComponentPrimitiveType), 8U);
124 dstArray->AsDoubleArray()->Memmove(dstPos, srcArray->AsDoubleArray(), srcPos, count);
125 return;
Ian Rogersef7d42f2014-01-06 12:55:46 -0800126 case Primitive::kPrimNot: {
127 mirror::ObjectArray<mirror::Object>* dstObjArray = dstArray->AsObjectArray<mirror::Object>();
128 mirror::ObjectArray<mirror::Object>* srcObjArray = srcArray->AsObjectArray<mirror::Object>();
129 dstObjArray->AssignableMemmove(dstPos, srcObjArray, srcPos, count);
130 return;
Elliott Hughesab3530d2012-01-09 16:04:56 -0800131 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800132 default:
David Sehr709b0702016-10-13 09:12:37 -0700133 LOG(FATAL) << "Unknown array type: " << srcArray->PrettyTypeOf();
Ian Rogers2c4257b2014-10-24 14:20:06 -0700134 UNREACHABLE();
Elliott Hughesbf86d042011-08-31 17:53:14 -0700135 }
136 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800137 // If one of the arrays holds a primitive type the other array must hold the exact same type.
138 if (UNLIKELY((dstComponentPrimitiveType != Primitive::kPrimNot) ||
139 srcComponentType->IsPrimitive())) {
David Sehr709b0702016-10-13 09:12:37 -0700140 std::string srcType(srcArray->PrettyTypeOf());
141 std::string dstType(dstArray->PrettyTypeOf());
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +0000142 soa.Self()->ThrowNewExceptionF("Ljava/lang/ArrayStoreException;",
Ian Rogersef7d42f2014-01-06 12:55:46 -0800143 "Incompatible types: src=%s, dst=%s",
144 srcType.c_str(), dstType.c_str());
Elliott Hughesbf86d042011-08-31 17:53:14 -0700145 return;
146 }
Ian Rogersef7d42f2014-01-06 12:55:46 -0800147 // Arrays hold distinct types and so therefore can't alias - use memcpy instead of memmove.
Mathieu Chartierbc5a7952016-10-17 15:46:31 -0700148 ObjPtr<mirror::ObjectArray<mirror::Object>> dstObjArray =
149 dstArray->AsObjectArray<mirror::Object>();
150 ObjPtr<mirror::ObjectArray<mirror::Object>> srcObjArray =
151 srcArray->AsObjectArray<mirror::Object>();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800152 // If we're assigning into say Object[] then we don't need per element checks.
153 if (dstComponentType->IsAssignableFrom(srcComponentType)) {
154 dstObjArray->AssignableMemcpy(dstPos, srcObjArray, srcPos, count);
155 return;
156 }
Andreas Gampe85a098a2016-03-31 13:30:53 -0700157 // This code is never run under a transaction.
158 DCHECK(!Runtime::Current()->IsActiveTransaction());
159 dstObjArray->AssignableCheckingMemcpy<false>(dstPos, srcObjArray, srcPos, count, true);
Elliott Hughesbf86d042011-08-31 17:53:14 -0700160}
161
Emma Meersmand735fe42014-06-18 11:50:59 -0700162// Template to convert general array to that of its specific primitive type.
163template <typename T>
Mathieu Chartierbc5a7952016-10-17 15:46:31 -0700164inline ObjPtr<T> AsPrimitiveArray(ObjPtr<mirror::Array> array)
165 REQUIRES_SHARED(Locks::mutator_lock_) {
166 return ObjPtr<T>::DownCast(array);
Emma Meersmand735fe42014-06-18 11:50:59 -0700167}
168
169template <typename T, Primitive::Type kPrimType>
170inline void System_arraycopyTUnchecked(JNIEnv* env, jobject javaSrc, jint srcPos,
171 jobject javaDst, jint dstPos, jint count) {
Ian Rogers1eb512d2013-10-18 15:42:20 -0700172 ScopedFastNativeObjectAccess soa(env);
Mathieu Chartier0795f232016-09-27 18:43:30 -0700173 ObjPtr<mirror::Object> srcObject = soa.Decode<mirror::Object>(javaSrc);
174 ObjPtr<mirror::Object> dstObject = soa.Decode<mirror::Object>(javaDst);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800175 DCHECK(dstObject != nullptr);
Mathieu Chartierbc5a7952016-10-17 15:46:31 -0700176 ObjPtr<mirror::Array> srcArray = srcObject->AsArray();
177 ObjPtr<mirror::Array> dstArray = dstObject->AsArray();
Ian Rogersef7d42f2014-01-06 12:55:46 -0800178 DCHECK_GE(count, 0);
Ian Rogersef7d42f2014-01-06 12:55:46 -0800179 DCHECK_EQ(srcArray->GetClass(), dstArray->GetClass());
Emma Meersmand735fe42014-06-18 11:50:59 -0700180 DCHECK_EQ(srcArray->GetClass()->GetComponentType()->GetPrimitiveType(), kPrimType);
181 AsPrimitiveArray<T>(dstArray)->Memmove(dstPos, AsPrimitiveArray<T>(srcArray), srcPos, count);
182}
183
184static void System_arraycopyCharUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
185 jobject javaDst, jint dstPos, jint count) {
186 System_arraycopyTUnchecked<mirror::CharArray, Primitive::kPrimChar>(env, javaSrc, srcPos,
187 javaDst, dstPos, count);
188}
189
190static void System_arraycopyByteUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
191 jobject javaDst, jint dstPos, jint count) {
192 System_arraycopyTUnchecked<mirror::ByteArray, Primitive::kPrimByte>(env, javaSrc, srcPos,
193 javaDst, dstPos, count);
194}
195
196static void System_arraycopyShortUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
197 jobject javaDst, jint dstPos, jint count) {
198 System_arraycopyTUnchecked<mirror::ShortArray, Primitive::kPrimShort>(env, javaSrc, srcPos,
199 javaDst, dstPos, count);
200}
201
202static void System_arraycopyIntUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
203 jobject javaDst, jint dstPos, jint count) {
204 System_arraycopyTUnchecked<mirror::IntArray, Primitive::kPrimInt>(env, javaSrc, srcPos,
205 javaDst, dstPos, count);
206}
207
208static void System_arraycopyLongUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
209 jobject javaDst, jint dstPos, jint count) {
210 System_arraycopyTUnchecked<mirror::LongArray, Primitive::kPrimLong>(env, javaSrc, srcPos,
211 javaDst, dstPos, count);
212}
213
214static void System_arraycopyFloatUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
215 jobject javaDst, jint dstPos, jint count) {
216 System_arraycopyTUnchecked<mirror::FloatArray, Primitive::kPrimFloat>(env, javaSrc, srcPos,
217 javaDst, dstPos, count);
218}
219
220static void System_arraycopyDoubleUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
221 jobject javaDst, jint dstPos, jint count) {
222 System_arraycopyTUnchecked<mirror::DoubleArray, Primitive::kPrimDouble>(env, javaSrc, srcPos,
223 javaDst, dstPos, count);
224}
225
226static void System_arraycopyBooleanUnchecked(JNIEnv* env, jclass, jobject javaSrc, jint srcPos,
227 jobject javaDst, jint dstPos, jint count) {
228 System_arraycopyTUnchecked<mirror::BooleanArray, Primitive::kPrimBoolean>(env, javaSrc, srcPos,
229 javaDst, dstPos, count);
Hiroshi Yamauchif38ea802013-08-27 13:04:26 -0700230}
231
Elliott Hughes0512f022012-03-15 22:10:52 -0700232static JNINativeMethod gMethods[] = {
Igor Murashkin3b6f4402017-02-16 16:13:17 -0800233 FAST_NATIVE_METHOD(System, arraycopy, "(Ljava/lang/Object;ILjava/lang/Object;II)V"),
234 FAST_NATIVE_METHOD(System, arraycopyCharUnchecked, "([CI[CII)V"),
235 FAST_NATIVE_METHOD(System, arraycopyByteUnchecked, "([BI[BII)V"),
236 FAST_NATIVE_METHOD(System, arraycopyShortUnchecked, "([SI[SII)V"),
237 FAST_NATIVE_METHOD(System, arraycopyIntUnchecked, "([II[III)V"),
238 FAST_NATIVE_METHOD(System, arraycopyLongUnchecked, "([JI[JII)V"),
239 FAST_NATIVE_METHOD(System, arraycopyFloatUnchecked, "([FI[FII)V"),
240 FAST_NATIVE_METHOD(System, arraycopyDoubleUnchecked, "([DI[DII)V"),
241 FAST_NATIVE_METHOD(System, arraycopyBooleanUnchecked, "([ZI[ZII)V"),
Elliott Hughesbf86d042011-08-31 17:53:14 -0700242};
243
Elliott Hughesbf86d042011-08-31 17:53:14 -0700244void register_java_lang_System(JNIEnv* env) {
Elliott Hugheseac76672012-05-24 21:56:51 -0700245 REGISTER_NATIVE_METHODS("java/lang/System");
Elliott Hughesbf86d042011-08-31 17:53:14 -0700246}
247
248} // namespace art