blob: 5355267b070e5d0f49fac7e942b75866d3c26895 [file] [log] [blame]
Ian Rogers57b86d42012-03-27 16:05:41 -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
Mathieu Chartiere401d142015-04-22 13:56:20 -070017#include "art_method-inl.h"
Andreas Gampe8228cdf2017-05-30 15:03:54 -070018#include "base/callee_save_type.h"
Ian Rogers57b86d42012-03-27 16:05:41 -070019#include "callee_save_frame.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080020#include "class_linker-inl.h"
Vladimir Markoaad75c62016-10-03 08:46:48 +000021#include "class_table-inl.h"
Ian Rogersfa46d3e2013-05-15 00:16:04 -070022#include "dex_file-inl.h"
Andreas Gampea5b09a62016-11-17 15:21:22 -080023#include "dex_file_types.h"
Andreas Gampe8cf9cb32017-07-19 09:28:38 -070024#include "entrypoints/entrypoint_utils-inl.h"
Vladimir Markoaad75c62016-10-03 08:46:48 +000025#include "gc/heap.h"
26#include "mirror/class-inl.h"
27#include "mirror/class_loader.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070028#include "mirror/object-inl.h"
Andreas Gampe8cf9cb32017-07-19 09:28:38 -070029#include "mirror/object_array-inl.h"
Vladimir Markoaad75c62016-10-03 08:46:48 +000030#include "oat_file.h"
31#include "runtime.h"
Ian Rogers57b86d42012-03-27 16:05:41 -070032
33namespace art {
34
Vladimir Marko6bec91c2017-01-09 15:03:12 +000035static inline void BssWriteBarrier(ArtMethod* outer_method) REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartiera1467d02017-02-22 09:22:50 -080036 // For AOT code, we need a write barrier for the class loader that holds the
37 // GC roots in the .bss.
38 const DexFile* dex_file = outer_method->GetDexFile();
Vladimir Marko6bec91c2017-01-09 15:03:12 +000039 if (dex_file != nullptr &&
40 dex_file->GetOatDexFile() != nullptr &&
41 !dex_file->GetOatDexFile()->GetOatFile()->GetBssGcRoots().empty()) {
Vladimir Marko9b03cb42017-02-16 16:37:03 +000042 ObjPtr<mirror::ClassLoader> class_loader = outer_method->GetClassLoader();
43 if (kIsDebugBuild) {
44 ClassTable* class_table =
45 Runtime::Current()->GetClassLinker()->ClassTableForClassLoader(class_loader);
46 CHECK(class_table != nullptr &&
47 !class_table->InsertOatFile(dex_file->GetOatDexFile()->GetOatFile()))
Vladimir Marko1998cd02017-01-13 13:02:58 +000048 << "Oat file with .bss GC roots was not registered in class table: "
49 << dex_file->GetOatDexFile()->GetOatFile()->GetLocation();
Vladimir Marko9b03cb42017-02-16 16:37:03 +000050 }
Mathieu Chartiera1467d02017-02-22 09:22:50 -080051 if (class_loader != nullptr) {
52 // Note that we emit the barrier before the compiled code stores the String or Class
53 // as a GC root. This is OK as there is no suspend point point in between.
54 Runtime::Current()->GetHeap()->WriteBarrierEveryFieldOf(class_loader);
55 } else {
56 Runtime::Current()->GetClassLinker()->WriteBarrierForBootOatFileBssRoots(
57 dex_file->GetOatDexFile()->GetOatFile());
Vladimir Marko1998cd02017-01-13 13:02:58 +000058 }
Vladimir Marko6bec91c2017-01-09 15:03:12 +000059 }
60}
61
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +010062extern "C" mirror::Class* artInitializeStaticStorageFromCode(uint32_t type_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070063 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerse2645d32012-04-11 14:42:42 -070064 // Called to ensure static storage base is initialized for direct static field reads and writes.
65 // A class may be accessing another class' fields when it doesn't have access, as access has been
66 // given by inheritance.
Ian Rogers1d8cdbc2014-09-22 22:51:09 -070067 ScopedQuickEntrypointChecks sqec(self);
Mingyao Yang0a87a652017-04-12 13:43:15 -070068 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(
69 self, CalleeSaveType::kSaveEverythingForClinit);
Vladimir Marko6bec91c2017-01-09 15:03:12 +000070 ArtMethod* caller = caller_and_outer.caller;
71 mirror::Class* result =
72 ResolveVerifyAndClinit(dex::TypeIndex(type_idx), caller, self, true, false);
73 if (LIKELY(result != nullptr)) {
74 BssWriteBarrier(caller_and_outer.outer_method);
75 }
76 return result;
Ian Rogers57b86d42012-03-27 16:05:41 -070077}
78
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +010079extern "C" mirror::Class* artInitializeTypeFromCode(uint32_t type_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070080 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerse2645d32012-04-11 14:42:42 -070081 // Called when method->dex_cache_resolved_types_[] misses.
Ian Rogers1d8cdbc2014-09-22 22:51:09 -070082 ScopedQuickEntrypointChecks sqec(self);
Mingyao Yang0a87a652017-04-12 13:43:15 -070083 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(
84 self, CalleeSaveType::kSaveEverythingForClinit);
Vladimir Marko6bec91c2017-01-09 15:03:12 +000085 ArtMethod* caller = caller_and_outer.caller;
86 mirror::Class* result =
87 ResolveVerifyAndClinit(dex::TypeIndex(type_idx), caller, self, false, false);
88 if (LIKELY(result != nullptr)) {
89 BssWriteBarrier(caller_and_outer.outer_method);
90 }
91 return result;
Ian Rogers57b86d42012-03-27 16:05:41 -070092}
93
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +010094extern "C" mirror::Class* artInitializeTypeAndVerifyAccessFromCode(uint32_t type_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070095 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogers57b86d42012-03-27 16:05:41 -070096 // Called when caller isn't guaranteed to have access to a type and the dex cache may be
Ian Rogerse2645d32012-04-11 14:42:42 -070097 // unpopulated.
Ian Rogers1d8cdbc2014-09-22 22:51:09 -070098 ScopedQuickEntrypointChecks sqec(self);
Andreas Gampe8228cdf2017-05-30 15:03:54 -070099 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(self,
100 CalleeSaveType::kSaveEverything);
Vladimir Marko6bec91c2017-01-09 15:03:12 +0000101 ArtMethod* caller = caller_and_outer.caller;
102 mirror::Class* result =
103 ResolveVerifyAndClinit(dex::TypeIndex(type_idx), caller, self, false, true);
104 if (LIKELY(result != nullptr)) {
105 BssWriteBarrier(caller_and_outer.outer_method);
106 }
107 return result;
Ian Rogers57b86d42012-03-27 16:05:41 -0700108}
109
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +0100110extern "C" mirror::String* artResolveStringFromCode(int32_t string_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700111 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogers1d8cdbc2014-09-22 22:51:09 -0700112 ScopedQuickEntrypointChecks sqec(self);
Andreas Gampe8228cdf2017-05-30 15:03:54 -0700113 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(self,
114 CalleeSaveType::kSaveEverything);
Vladimir Marko6bec91c2017-01-09 15:03:12 +0000115 ArtMethod* caller = caller_and_outer.caller;
Andreas Gampe8a0128a2016-11-28 07:38:35 -0800116 mirror::String* result = ResolveStringFromCode(caller, dex::StringIndex(string_idx));
Vladimir Markoaad75c62016-10-03 08:46:48 +0000117 if (LIKELY(result != nullptr)) {
Vladimir Marko6bec91c2017-01-09 15:03:12 +0000118 BssWriteBarrier(caller_and_outer.outer_method);
Vladimir Markoaad75c62016-10-03 08:46:48 +0000119 }
120 return result;
Ian Rogers57b86d42012-03-27 16:05:41 -0700121}
122
123} // namespace art