blob: 970593d119c12302cb90f445bad165f3bd200e60 [file] [log] [blame]
Ian Rogers87e552d2012-08-31 15:54:48 -07001/*
2 * Copyright (C) 2012 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 "common_throws.h"
18
Ian Rogers22d5e732014-07-15 22:23:51 -070019#include <sstream>
20
Elliott Hughes07ed66b2012-12-12 18:34:25 -080021#include "base/logging.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080022#include "class_linker-inl.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070023#include "dex_file-inl.h"
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +020024#include "dex_instruction-inl.h"
Ian Rogers87e552d2012-08-31 15:54:48 -070025#include "invoke_type.h"
Brian Carlstromea46f952013-07-30 01:26:50 -070026#include "mirror/art_method-inl.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070027#include "mirror/class-inl.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080028#include "mirror/object-inl.h"
29#include "mirror/object_array-inl.h"
Ian Rogers87e552d2012-08-31 15:54:48 -070030#include "thread.h"
Sebastien Hertz2d6ba512013-05-17 11:31:37 +020031#include "verifier/method_verifier.h"
Ian Rogers87e552d2012-08-31 15:54:48 -070032
Ian Rogers87e552d2012-08-31 15:54:48 -070033namespace art {
34
Ian Rogersef7d42f2014-01-06 12:55:46 -080035static void AddReferrerLocation(std::ostream& os, mirror::Class* referrer)
Ian Rogersb726dcb2012-09-05 08:57:23 -070036 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers87e552d2012-08-31 15:54:48 -070037 if (referrer != NULL) {
Mathieu Chartierf8322842014-05-16 10:59:25 -070038 std::string location(referrer->GetLocation());
Ian Rogers87e552d2012-08-31 15:54:48 -070039 if (!location.empty()) {
40 os << " (declaration of '" << PrettyDescriptor(referrer)
41 << "' appears in " << location << ")";
42 }
43 }
44}
45
Ian Rogers62d6c772013-02-27 08:32:07 -080046static void ThrowException(const ThrowLocation* throw_location, const char* exception_descriptor,
Ian Rogersef7d42f2014-01-06 12:55:46 -080047 mirror::Class* referrer, const char* fmt, va_list* args = NULL)
Ian Rogers62d6c772013-02-27 08:32:07 -080048 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers87e552d2012-08-31 15:54:48 -070049 std::ostringstream msg;
Ian Rogers62d6c772013-02-27 08:32:07 -080050 if (args != NULL) {
51 std::string vmsg;
52 StringAppendV(&vmsg, fmt, *args);
53 msg << vmsg;
54 } else {
55 msg << fmt;
56 }
57 AddReferrerLocation(msg, referrer);
58 Thread* self = Thread::Current();
59 if (throw_location == NULL) {
60 ThrowLocation computed_throw_location = self->GetCurrentLocationForThrow();
61 self->ThrowNewException(computed_throw_location, exception_descriptor, msg.str().c_str());
62 } else {
63 self->ThrowNewException(*throw_location, exception_descriptor, msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -070064 }
65}
66
Andreas Gampe329d1882014-04-08 10:32:19 -070067static void ThrowWrappedException(const ThrowLocation* throw_location,
68 const char* exception_descriptor,
69 mirror::Class* referrer, const char* fmt, va_list* args = NULL)
70 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
71 std::ostringstream msg;
72 if (args != NULL) {
73 std::string vmsg;
74 StringAppendV(&vmsg, fmt, *args);
75 msg << vmsg;
76 } else {
77 msg << fmt;
78 }
79 AddReferrerLocation(msg, referrer);
80 Thread* self = Thread::Current();
81 if (throw_location == NULL) {
82 ThrowLocation computed_throw_location = self->GetCurrentLocationForThrow();
83 self->ThrowNewWrappedException(computed_throw_location, exception_descriptor, msg.str().c_str());
84 } else {
85 self->ThrowNewWrappedException(*throw_location, exception_descriptor, msg.str().c_str());
86 }
87}
88
Sebastien Hertz56adf602013-07-09 17:27:07 +020089// AbstractMethodError
90
Ian Rogersef7d42f2014-01-06 12:55:46 -080091void ThrowAbstractMethodError(mirror::ArtMethod* method) {
Sebastien Hertz56adf602013-07-09 17:27:07 +020092 ThrowException(NULL, "Ljava/lang/AbstractMethodError;", NULL,
93 StringPrintf("abstract method \"%s\"",
94 PrettyMethod(method).c_str()).c_str());
95}
96
Ian Rogers62d6c772013-02-27 08:32:07 -080097// ArithmeticException
98
Sebastien Hertz0a3b8632013-06-26 11:16:01 +020099void ThrowArithmeticExceptionDivideByZero() {
Ian Rogers62d6c772013-02-27 08:32:07 -0800100 ThrowException(NULL, "Ljava/lang/ArithmeticException;", NULL, "divide by zero");
101}
102
103// ArrayIndexOutOfBoundsException
104
105void ThrowArrayIndexOutOfBoundsException(int index, int length) {
106 ThrowException(NULL, "Ljava/lang/ArrayIndexOutOfBoundsException;", NULL,
107 StringPrintf("length=%d; index=%d", length, index).c_str());
108}
109
110// ArrayStoreException
111
Ian Rogersef7d42f2014-01-06 12:55:46 -0800112void ThrowArrayStoreException(mirror::Class* element_class, mirror::Class* array_class) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800113 ThrowException(NULL, "Ljava/lang/ArrayStoreException;", NULL,
114 StringPrintf("%s cannot be stored in an array of type %s",
115 PrettyDescriptor(element_class).c_str(),
116 PrettyDescriptor(array_class).c_str()).c_str());
117}
118
119// ClassCastException
120
Ian Rogersef7d42f2014-01-06 12:55:46 -0800121void ThrowClassCastException(mirror::Class* dest_type, mirror::Class* src_type) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800122 ThrowException(NULL, "Ljava/lang/ClassCastException;", NULL,
123 StringPrintf("%s cannot be cast to %s",
124 PrettyDescriptor(src_type).c_str(),
125 PrettyDescriptor(dest_type).c_str()).c_str());
126}
127
128void ThrowClassCastException(const ThrowLocation* throw_location, const char* msg) {
129 ThrowException(throw_location, "Ljava/lang/ClassCastException;", NULL, msg);
130}
131
132// ClassCircularityError
133
134void ThrowClassCircularityError(mirror::Class* c) {
135 std::ostringstream msg;
136 msg << PrettyDescriptor(c);
137 ThrowException(NULL, "Ljava/lang/ClassCircularityError;", c, msg.str().c_str());
138}
139
140// ClassFormatError
141
Ian Rogersef7d42f2014-01-06 12:55:46 -0800142void ThrowClassFormatError(mirror::Class* referrer, const char* fmt, ...) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800143 va_list args;
144 va_start(args, fmt);
145 ThrowException(NULL, "Ljava/lang/ClassFormatError;", referrer, fmt, &args);
146 va_end(args);}
147
Ian Rogers87e552d2012-08-31 15:54:48 -0700148// IllegalAccessError
149
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800150void ThrowIllegalAccessErrorClass(mirror::Class* referrer, mirror::Class* accessed) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700151 std::ostringstream msg;
Ian Rogersb726dcb2012-09-05 08:57:23 -0700152 msg << "Illegal class access: '" << PrettyDescriptor(referrer) << "' attempting to access '"
Ian Rogers87e552d2012-08-31 15:54:48 -0700153 << PrettyDescriptor(accessed) << "'";
Ian Rogers62d6c772013-02-27 08:32:07 -0800154 ThrowException(NULL, "Ljava/lang/IllegalAccessError;", referrer, msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700155}
156
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800157void ThrowIllegalAccessErrorClassForMethodDispatch(mirror::Class* referrer, mirror::Class* accessed,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800158 mirror::ArtMethod* called,
Ian Rogers87e552d2012-08-31 15:54:48 -0700159 InvokeType type) {
160 std::ostringstream msg;
Ian Rogersb726dcb2012-09-05 08:57:23 -0700161 msg << "Illegal class access ('" << PrettyDescriptor(referrer) << "' attempting to access '"
162 << PrettyDescriptor(accessed) << "') in attempt to invoke " << type
Ian Rogers87e552d2012-08-31 15:54:48 -0700163 << " method " << PrettyMethod(called).c_str();
Ian Rogers62d6c772013-02-27 08:32:07 -0800164 ThrowException(NULL, "Ljava/lang/IllegalAccessError;", referrer, msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700165}
166
Brian Carlstromea46f952013-07-30 01:26:50 -0700167void ThrowIllegalAccessErrorMethod(mirror::Class* referrer, mirror::ArtMethod* accessed) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700168 std::ostringstream msg;
169 msg << "Method '" << PrettyMethod(accessed) << "' is inaccessible to class '"
170 << PrettyDescriptor(referrer) << "'";
Ian Rogers62d6c772013-02-27 08:32:07 -0800171 ThrowException(NULL, "Ljava/lang/IllegalAccessError;", referrer, msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700172}
173
Brian Carlstromea46f952013-07-30 01:26:50 -0700174void ThrowIllegalAccessErrorField(mirror::Class* referrer, mirror::ArtField* accessed) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700175 std::ostringstream msg;
176 msg << "Field '" << PrettyField(accessed, false) << "' is inaccessible to class '"
177 << PrettyDescriptor(referrer) << "'";
Ian Rogers62d6c772013-02-27 08:32:07 -0800178 ThrowException(NULL, "Ljava/lang/IllegalAccessError;", referrer, msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700179}
180
Ian Rogersef7d42f2014-01-06 12:55:46 -0800181void ThrowIllegalAccessErrorFinalField(mirror::ArtMethod* referrer,
Brian Carlstromea46f952013-07-30 01:26:50 -0700182 mirror::ArtField* accessed) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700183 std::ostringstream msg;
184 msg << "Final field '" << PrettyField(accessed, false) << "' cannot be written to by method '"
185 << PrettyMethod(referrer) << "'";
Brian Carlstromea46f952013-07-30 01:26:50 -0700186 ThrowException(NULL, "Ljava/lang/IllegalAccessError;",
187 referrer != NULL ? referrer->GetClass() : NULL,
Ian Rogers62d6c772013-02-27 08:32:07 -0800188 msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700189}
190
Brian Carlstrom2ce745c2013-07-17 17:44:30 -0700191void ThrowIllegalAccessError(mirror::Class* referrer, const char* fmt, ...) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800192 va_list args;
193 va_start(args, fmt);
194 ThrowException(NULL, "Ljava/lang/IllegalAccessError;", referrer, fmt, &args);
195 va_end(args);
196}
197
Jeff Hao11d5d8f2014-03-26 15:08:20 -0700198// IllegalAccessException
199
200void ThrowIllegalAccessException(const ThrowLocation* throw_location, const char* msg) {
201 ThrowException(throw_location, "Ljava/lang/IllegalAccessException;", NULL, msg);
202}
203
Ian Rogers62d6c772013-02-27 08:32:07 -0800204// IllegalArgumentException
205
206void ThrowIllegalArgumentException(const ThrowLocation* throw_location, const char* msg) {
207 ThrowException(throw_location, "Ljava/lang/IllegalArgumentException;", NULL, msg);
208}
209
210
Ian Rogers87e552d2012-08-31 15:54:48 -0700211// IncompatibleClassChangeError
212
213void ThrowIncompatibleClassChangeError(InvokeType expected_type, InvokeType found_type,
Brian Carlstromea46f952013-07-30 01:26:50 -0700214 mirror::ArtMethod* method,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800215 mirror::ArtMethod* referrer) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700216 std::ostringstream msg;
217 msg << "The method '" << PrettyMethod(method) << "' was expected to be of type "
218 << expected_type << " but instead was found to be of type " << found_type;
Ian Rogers62d6c772013-02-27 08:32:07 -0800219 ThrowException(NULL, "Ljava/lang/IncompatibleClassChangeError;",
220 referrer != NULL ? referrer->GetClass() : NULL,
221 msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700222}
223
Ian Rogersef7d42f2014-01-06 12:55:46 -0800224void ThrowIncompatibleClassChangeErrorClassForInterfaceDispatch(mirror::ArtMethod* interface_method,
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800225 mirror::Object* this_object,
Ian Rogersef7d42f2014-01-06 12:55:46 -0800226 mirror::ArtMethod* referrer) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700227 // Referrer is calling interface_method on this_object, however, the interface_method isn't
228 // implemented by this_object.
229 CHECK(this_object != NULL);
230 std::ostringstream msg;
231 msg << "Class '" << PrettyDescriptor(this_object->GetClass())
232 << "' does not implement interface '"
233 << PrettyDescriptor(interface_method->GetDeclaringClass())
234 << "' in call to '" << PrettyMethod(interface_method) << "'";
Ian Rogers62d6c772013-02-27 08:32:07 -0800235 ThrowException(NULL, "Ljava/lang/IncompatibleClassChangeError;",
236 referrer != NULL ? referrer->GetClass() : NULL,
237 msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700238}
239
Ian Rogersef7d42f2014-01-06 12:55:46 -0800240void ThrowIncompatibleClassChangeErrorField(mirror::ArtField* resolved_field, bool is_static,
241 mirror::ArtMethod* referrer) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700242 std::ostringstream msg;
243 msg << "Expected '" << PrettyField(resolved_field) << "' to be a "
Ian Rogersb726dcb2012-09-05 08:57:23 -0700244 << (is_static ? "static" : "instance") << " field" << " rather than a "
245 << (is_static ? "instance" : "static") << " field";
Ian Rogers62d6c772013-02-27 08:32:07 -0800246 ThrowException(NULL, "Ljava/lang/IncompatibleClassChangeError;", referrer->GetClass(),
247 msg.str().c_str());
248}
249
Ian Rogersef7d42f2014-01-06 12:55:46 -0800250void ThrowIncompatibleClassChangeError(mirror::Class* referrer, const char* fmt, ...) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800251 va_list args;
252 va_start(args, fmt);
253 ThrowException(NULL, "Ljava/lang/IncompatibleClassChangeError;", referrer, fmt, &args);
254 va_end(args);
255}
256
Ian Rogers8d31bbd2013-10-13 10:44:14 -0700257// IOException
258
259void ThrowIOException(const char* fmt, ...) {
260 va_list args;
261 va_start(args, fmt);
262 ThrowException(NULL, "Ljava/io/IOException;", NULL, fmt, &args);
263 va_end(args);
264}
265
Andreas Gampe329d1882014-04-08 10:32:19 -0700266void ThrowWrappedIOException(const char* fmt, ...) {
267 va_list args;
268 va_start(args, fmt);
269 ThrowWrappedException(NULL, "Ljava/io/IOException;", NULL, fmt, &args);
270 va_end(args);
271}
272
Ian Rogers62d6c772013-02-27 08:32:07 -0800273// LinkageError
274
Ian Rogersef7d42f2014-01-06 12:55:46 -0800275void ThrowLinkageError(mirror::Class* referrer, const char* fmt, ...) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800276 va_list args;
277 va_start(args, fmt);
278 ThrowException(NULL, "Ljava/lang/LinkageError;", referrer, fmt, &args);
279 va_end(args);
280}
281
282// NegativeArraySizeException
283
284void ThrowNegativeArraySizeException(int size) {
Brian Carlstromea46f952013-07-30 01:26:50 -0700285 ThrowException(NULL, "Ljava/lang/NegativeArraySizeException;", NULL,
286 StringPrintf("%d", size).c_str());
Ian Rogers62d6c772013-02-27 08:32:07 -0800287}
288
289void ThrowNegativeArraySizeException(const char* msg) {
290 ThrowException(NULL, "Ljava/lang/NegativeArraySizeException;", NULL, msg);
291}
292
293// NoSuchFieldError
294
295void ThrowNoSuchFieldError(const StringPiece& scope, mirror::Class* c,
296 const StringPiece& type, const StringPiece& name)
297 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800298 std::ostringstream msg;
299 msg << "No " << scope << "field " << name << " of type " << type
Mathieu Chartierf8322842014-05-16 10:59:25 -0700300 << " in class " << c->GetDescriptor() << " or its superclasses";
Ian Rogers62d6c772013-02-27 08:32:07 -0800301 ThrowException(NULL, "Ljava/lang/NoSuchFieldError;", c, msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700302}
303
304// NoSuchMethodError
305
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800306void ThrowNoSuchMethodError(InvokeType type, mirror::Class* c, const StringPiece& name,
Ian Rogersd91d6d62013-09-25 20:26:14 -0700307 const Signature& signature) {
Ian Rogers87e552d2012-08-31 15:54:48 -0700308 std::ostringstream msg;
Ian Rogers87e552d2012-08-31 15:54:48 -0700309 msg << "No " << type << " method " << name << signature
Mathieu Chartierf8322842014-05-16 10:59:25 -0700310 << " in class " << c->GetDescriptor() << " or its super classes";
Ian Rogers62d6c772013-02-27 08:32:07 -0800311 ThrowException(NULL, "Ljava/lang/NoSuchMethodError;", c, msg.str().c_str());
Ian Rogers87e552d2012-08-31 15:54:48 -0700312}
313
Ian Rogers62d6c772013-02-27 08:32:07 -0800314void ThrowNoSuchMethodError(uint32_t method_idx) {
315 Thread* self = Thread::Current();
316 ThrowLocation throw_location = self->GetCurrentLocationForThrow();
317 mirror::DexCache* dex_cache = throw_location.GetMethod()->GetDeclaringClass()->GetDexCache();
Ian Rogers4445a7e2012-10-05 17:19:13 -0700318 const DexFile& dex_file = *dex_cache->GetDexFile();
Ian Rogers87e552d2012-08-31 15:54:48 -0700319 std::ostringstream msg;
320 msg << "No method '" << PrettyMethod(method_idx, dex_file, true) << "'";
Ian Rogers62d6c772013-02-27 08:32:07 -0800321 ThrowException(&throw_location, "Ljava/lang/NoSuchMethodError;",
322 throw_location.GetMethod()->GetDeclaringClass(), msg.str().c_str());
323}
324
325// NullPointerException
326
327void ThrowNullPointerExceptionForFieldAccess(const ThrowLocation& throw_location,
Brian Carlstromea46f952013-07-30 01:26:50 -0700328 mirror::ArtField* field, bool is_read) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800329 std::ostringstream msg;
330 msg << "Attempt to " << (is_read ? "read from" : "write to")
331 << " field '" << PrettyField(field, true) << "' on a null object reference";
332 ThrowException(&throw_location, "Ljava/lang/NullPointerException;", NULL, msg.str().c_str());
333}
334
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200335static void ThrowNullPointerExceptionForMethodAccessImpl(const ThrowLocation& throw_location,
336 uint32_t method_idx,
337 const DexFile& dex_file,
338 InvokeType type)
339 SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800340 std::ostringstream msg;
341 msg << "Attempt to invoke " << type << " method '"
342 << PrettyMethod(method_idx, dex_file, true) << "' on a null object reference";
343 ThrowException(&throw_location, "Ljava/lang/NullPointerException;", NULL, msg.str().c_str());
344}
345
Brian Carlstromea46f952013-07-30 01:26:50 -0700346void ThrowNullPointerExceptionForMethodAccess(const ThrowLocation& throw_location,
347 uint32_t method_idx,
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200348 InvokeType type) {
349 mirror::DexCache* dex_cache = throw_location.GetMethod()->GetDeclaringClass()->GetDexCache();
350 const DexFile& dex_file = *dex_cache->GetDexFile();
351 ThrowNullPointerExceptionForMethodAccessImpl(throw_location, method_idx,
352 dex_file, type);
353}
354
355void ThrowNullPointerExceptionForMethodAccess(const ThrowLocation& throw_location,
Brian Carlstromea46f952013-07-30 01:26:50 -0700356 mirror::ArtMethod* method,
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200357 InvokeType type) {
358 mirror::DexCache* dex_cache = method->GetDeclaringClass()->GetDexCache();
359 const DexFile& dex_file = *dex_cache->GetDexFile();
360 ThrowNullPointerExceptionForMethodAccessImpl(throw_location, method->GetDexMethodIndex(),
361 dex_file, type);
362}
363
Ian Rogers62d6c772013-02-27 08:32:07 -0800364void ThrowNullPointerExceptionFromDexPC(const ThrowLocation& throw_location) {
Mathieu Chartierbfd9a432014-05-21 17:43:44 -0700365 const DexFile::CodeItem* code = throw_location.GetMethod()->GetCodeItem();
Ian Rogers62d6c772013-02-27 08:32:07 -0800366 uint32_t throw_dex_pc = throw_location.GetDexPc();
367 CHECK_LT(throw_dex_pc, code->insns_size_in_code_units_);
368 const Instruction* instr = Instruction::At(&code->insns_[throw_dex_pc]);
Ian Rogers62d6c772013-02-27 08:32:07 -0800369 switch (instr->Opcode()) {
370 case Instruction::INVOKE_DIRECT:
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200371 ThrowNullPointerExceptionForMethodAccess(throw_location, instr->VRegB_35c(), kDirect);
372 break;
Ian Rogers62d6c772013-02-27 08:32:07 -0800373 case Instruction::INVOKE_DIRECT_RANGE:
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200374 ThrowNullPointerExceptionForMethodAccess(throw_location, instr->VRegB_3rc(), kDirect);
Ian Rogers62d6c772013-02-27 08:32:07 -0800375 break;
376 case Instruction::INVOKE_VIRTUAL:
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200377 ThrowNullPointerExceptionForMethodAccess(throw_location, instr->VRegB_35c(), kVirtual);
378 break;
Ian Rogers62d6c772013-02-27 08:32:07 -0800379 case Instruction::INVOKE_VIRTUAL_RANGE:
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200380 ThrowNullPointerExceptionForMethodAccess(throw_location, instr->VRegB_3rc(), kVirtual);
Ian Rogers62d6c772013-02-27 08:32:07 -0800381 break;
382 case Instruction::INVOKE_INTERFACE:
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200383 ThrowNullPointerExceptionForMethodAccess(throw_location, instr->VRegB_35c(), kInterface);
384 break;
Ian Rogers62d6c772013-02-27 08:32:07 -0800385 case Instruction::INVOKE_INTERFACE_RANGE:
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200386 ThrowNullPointerExceptionForMethodAccess(throw_location, instr->VRegB_3rc(), kInterface);
Ian Rogers62d6c772013-02-27 08:32:07 -0800387 break;
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200388 case Instruction::INVOKE_VIRTUAL_QUICK:
389 case Instruction::INVOKE_VIRTUAL_RANGE_QUICK: {
390 // Since we replaced the method index, we ask the verifier to tell us which
391 // method is invoked at this location.
Brian Carlstromea46f952013-07-30 01:26:50 -0700392 mirror::ArtMethod* method =
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200393 verifier::MethodVerifier::FindInvokedMethodAtDexPc(throw_location.GetMethod(),
394 throw_location.GetDexPc());
395 if (method != NULL) {
396 // NPE with precise message.
397 ThrowNullPointerExceptionForMethodAccess(throw_location, method, kVirtual);
398 } else {
399 // NPE with imprecise message.
400 ThrowNullPointerException(&throw_location,
401 "Attempt to invoke a virtual method on a null object reference");
402 }
403 break;
404 }
Ian Rogers62d6c772013-02-27 08:32:07 -0800405 case Instruction::IGET:
406 case Instruction::IGET_WIDE:
407 case Instruction::IGET_OBJECT:
408 case Instruction::IGET_BOOLEAN:
409 case Instruction::IGET_BYTE:
410 case Instruction::IGET_CHAR:
411 case Instruction::IGET_SHORT: {
Brian Carlstromea46f952013-07-30 01:26:50 -0700412 mirror::ArtField* field =
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200413 Runtime::Current()->GetClassLinker()->ResolveField(instr->VRegC_22c(),
Ian Rogers62d6c772013-02-27 08:32:07 -0800414 throw_location.GetMethod(), false);
415 ThrowNullPointerExceptionForFieldAccess(throw_location, field, true /* read */);
416 break;
417 }
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200418 case Instruction::IGET_QUICK:
419 case Instruction::IGET_WIDE_QUICK:
420 case Instruction::IGET_OBJECT_QUICK: {
421 // Since we replaced the field index, we ask the verifier to tell us which
422 // field is accessed at this location.
Brian Carlstromea46f952013-07-30 01:26:50 -0700423 mirror::ArtField* field =
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200424 verifier::MethodVerifier::FindAccessedFieldAtDexPc(throw_location.GetMethod(),
425 throw_location.GetDexPc());
426 if (field != NULL) {
427 // NPE with precise message.
428 ThrowNullPointerExceptionForFieldAccess(throw_location, field, true /* read */);
429 } else {
430 // NPE with imprecise message.
431 ThrowNullPointerException(&throw_location,
432 "Attempt to read from a field on a null object reference");
433 }
434 break;
435 }
Ian Rogers62d6c772013-02-27 08:32:07 -0800436 case Instruction::IPUT:
437 case Instruction::IPUT_WIDE:
438 case Instruction::IPUT_OBJECT:
439 case Instruction::IPUT_BOOLEAN:
440 case Instruction::IPUT_BYTE:
441 case Instruction::IPUT_CHAR:
442 case Instruction::IPUT_SHORT: {
Brian Carlstromea46f952013-07-30 01:26:50 -0700443 mirror::ArtField* field =
Sebastien Hertz75b2a4a2013-05-21 09:25:10 +0200444 Runtime::Current()->GetClassLinker()->ResolveField(instr->VRegC_22c(),
Ian Rogers62d6c772013-02-27 08:32:07 -0800445 throw_location.GetMethod(), false);
446 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false /* write */);
447 break;
448 }
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200449 case Instruction::IPUT_QUICK:
450 case Instruction::IPUT_WIDE_QUICK:
451 case Instruction::IPUT_OBJECT_QUICK: {
452 // Since we replaced the field index, we ask the verifier to tell us which
453 // field is accessed at this location.
Brian Carlstromea46f952013-07-30 01:26:50 -0700454 mirror::ArtField* field =
Sebastien Hertz2d6ba512013-05-17 11:31:37 +0200455 verifier::MethodVerifier::FindAccessedFieldAtDexPc(throw_location.GetMethod(),
456 throw_location.GetDexPc());
457 if (field != NULL) {
458 // NPE with precise message.
459 ThrowNullPointerExceptionForFieldAccess(throw_location, field, false /* write */);
460 } else {
461 // NPE with imprecise message.
462 ThrowNullPointerException(&throw_location,
463 "Attempt to write to a field on a null object reference");
464 }
465 break;
466 }
Ian Rogers62d6c772013-02-27 08:32:07 -0800467 case Instruction::AGET:
468 case Instruction::AGET_WIDE:
469 case Instruction::AGET_OBJECT:
470 case Instruction::AGET_BOOLEAN:
471 case Instruction::AGET_BYTE:
472 case Instruction::AGET_CHAR:
473 case Instruction::AGET_SHORT:
474 ThrowException(&throw_location, "Ljava/lang/NullPointerException;", NULL,
475 "Attempt to read from null array");
476 break;
477 case Instruction::APUT:
478 case Instruction::APUT_WIDE:
479 case Instruction::APUT_OBJECT:
480 case Instruction::APUT_BOOLEAN:
481 case Instruction::APUT_BYTE:
482 case Instruction::APUT_CHAR:
483 case Instruction::APUT_SHORT:
484 ThrowException(&throw_location, "Ljava/lang/NullPointerException;", NULL,
485 "Attempt to write to null array");
486 break;
487 case Instruction::ARRAY_LENGTH:
488 ThrowException(&throw_location, "Ljava/lang/NullPointerException;", NULL,
489 "Attempt to get length of null array");
490 break;
491 default: {
492 // TODO: We should have covered all the cases where we expect a NPE above, this
493 // message/logging is so we can improve any cases we've missed in the future.
494 const DexFile& dex_file =
495 *throw_location.GetMethod()->GetDeclaringClass()->GetDexCache()->GetDexFile();
496 ThrowException(&throw_location, "Ljava/lang/NullPointerException;", NULL,
497 StringPrintf("Null pointer exception during instruction '%s'",
498 instr->DumpString(&dex_file).c_str()).c_str());
499 break;
500 }
501 }
502}
503
504void ThrowNullPointerException(const ThrowLocation* throw_location, const char* msg) {
505 ThrowException(throw_location, "Ljava/lang/NullPointerException;", NULL, msg);
506}
507
508// RuntimeException
509
510void ThrowRuntimeException(const char* fmt, ...) {
511 va_list args;
512 va_start(args, fmt);
513 ThrowException(NULL, "Ljava/lang/RuntimeException;", NULL, fmt, &args);
514 va_end(args);
515}
516
517// VerifyError
518
Ian Rogersef7d42f2014-01-06 12:55:46 -0800519void ThrowVerifyError(mirror::Class* referrer, const char* fmt, ...) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800520 va_list args;
521 va_start(args, fmt);
522 ThrowException(NULL, "Ljava/lang/VerifyError;", referrer, fmt, &args);
523 va_end(args);
Ian Rogers87e552d2012-08-31 15:54:48 -0700524}
525
526} // namespace art