1 /*
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 "dex_instruction.h"
18 #include "entrypoints/entrypoint_utils-inl.h"
19 #include "mirror/art_method-inl.h"
20 #include "mirror/object-inl.h"
21 
22 namespace art {
23 
art_portable_throw_div_zero_from_code()24 extern "C" void art_portable_throw_div_zero_from_code() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
25   ThrowArithmeticExceptionDivideByZero();
26 }
27 
art_portable_throw_array_bounds_from_code(int32_t index,int32_t length)28 extern "C" void art_portable_throw_array_bounds_from_code(int32_t index, int32_t length)
29     SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
30   ThrowArrayIndexOutOfBoundsException(index, length);
31 }
32 
art_portable_throw_no_such_method_from_code(int32_t method_idx)33 extern "C" void art_portable_throw_no_such_method_from_code(int32_t method_idx)
34     SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
35   ThrowNoSuchMethodError(method_idx);
36 }
37 
art_portable_throw_null_pointer_exception_from_code(uint32_t dex_pc)38 extern "C" void art_portable_throw_null_pointer_exception_from_code(uint32_t dex_pc)
39     SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
40   // TODO: remove dex_pc argument from caller.
41   UNUSED(dex_pc);
42   Thread* self = Thread::Current();
43   ThrowLocation throw_location = self->GetCurrentLocationForThrow();
44   ThrowNullPointerExceptionFromDexPC(throw_location);
45 }
46 
art_portable_throw_stack_overflow_from_code()47 extern "C" void art_portable_throw_stack_overflow_from_code() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
48   ThrowStackOverflowError(Thread::Current());
49 }
50 
art_portable_throw_exception_from_code(mirror::Throwable * exception)51 extern "C" void art_portable_throw_exception_from_code(mirror::Throwable* exception)
52     SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
53   Thread* self = Thread::Current();
54   ThrowLocation throw_location = self->GetCurrentLocationForThrow();
55   if (exception == NULL) {
56     ThrowNullPointerException(NULL, "throw with null exception");
57   } else {
58     self->SetException(throw_location, exception);
59   }
60 }
61 
art_portable_get_and_clear_exception(Thread * self)62 extern "C" void* art_portable_get_and_clear_exception(Thread* self)
63     SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
64   DCHECK(self->IsExceptionPending());
65   // TODO: make this inline.
66   mirror::Throwable* exception = self->GetException(NULL);
67   self->ClearException();
68   return exception;
69 }
70 
art_portable_find_catch_block_from_code(mirror::ArtMethod * current_method,uint32_t ti_offset)71 extern "C" int32_t art_portable_find_catch_block_from_code(mirror::ArtMethod* current_method,
72                                                            uint32_t ti_offset)
73     SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
74   Thread* self = Thread::Current();  // TODO: make an argument.
75   ThrowLocation throw_location;
76   mirror::Throwable* exception = self->GetException(&throw_location);
77   // Check for special deoptimization exception.
78   if (UNLIKELY(reinterpret_cast<intptr_t>(exception) == -1)) {
79     return -1;
80   }
81   mirror::Class* exception_type = exception->GetClass();
82   StackHandleScope<1> hs(self);
83   const DexFile::CodeItem* code_item = current_method->GetCodeItem();
84   DCHECK_LT(ti_offset, code_item->tries_size_);
85   const DexFile::TryItem* try_item = DexFile::GetTryItems(*code_item, ti_offset);
86 
87   int iter_index = 0;
88   int result = -1;
89   uint32_t catch_dex_pc = -1;
90   // Iterate over the catch handlers associated with dex_pc
91   for (CatchHandlerIterator it(*code_item, *try_item); it.HasNext(); it.Next()) {
92     uint16_t iter_type_idx = it.GetHandlerTypeIndex();
93     // Catch all case
94     if (iter_type_idx == DexFile::kDexNoIndex16) {
95       catch_dex_pc = it.GetHandlerAddress();
96       result = iter_index;
97       break;
98     }
99     // Does this catch exception type apply?
100     mirror::Class* iter_exception_type =
101         current_method->GetDexCacheResolvedType(iter_type_idx);
102     if (UNLIKELY(iter_exception_type == NULL)) {
103       // TODO: check, the verifier (class linker?) should take care of resolving all exception
104       //       classes early.
105       LOG(WARNING) << "Unresolved exception class when finding catch block: "
106           << current_method->GetTypeDescriptorFromTypeIdx(iter_type_idx);
107     } else if (iter_exception_type->IsAssignableFrom(exception_type)) {
108       catch_dex_pc = it.GetHandlerAddress();
109       result = iter_index;
110       break;
111     }
112     ++iter_index;
113   }
114   if (result != -1) {
115     // Handler found.
116     Runtime::Current()->GetInstrumentation()->ExceptionCaughtEvent(
117         self, throw_location, current_method, catch_dex_pc, exception);
118     // If the catch block has no move-exception then clear the exception for it.
119     const Instruction* first_catch_instr = Instruction::At(
120         &current_method->GetCodeItem()->insns_[catch_dex_pc]);
121     if (first_catch_instr->Opcode() != Instruction::MOVE_EXCEPTION) {
122       self->ClearException();
123     }
124   }
125   return result;
126 }
127 
128 }  // namespace art
129