blob: 355d7b3e2f730eb2bdf5060aaaec4d468155cde6 [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"
Ian Rogers57b86d42012-03-27 16:05:41 -070018#include "callee_save_frame.h"
Mingyao Yang98d1cc82014-05-15 17:02:16 -070019#include "entrypoints/entrypoint_utils-inl.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"
Vladimir Markoaad75c62016-10-03 08:46:48 +000024#include "gc/heap.h"
25#include "mirror/class-inl.h"
26#include "mirror/class_loader.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080027#include "mirror/object_array-inl.h"
Ian Rogers4f6ad8a2013-03-18 15:27:28 -070028#include "mirror/object-inl.h"
Vladimir Markoaad75c62016-10-03 08:46:48 +000029#include "oat_file.h"
30#include "runtime.h"
Ian Rogers57b86d42012-03-27 16:05:41 -070031
32namespace art {
33
Vladimir Marko6bec91c2017-01-09 15:03:12 +000034static inline void BssWriteBarrier(ArtMethod* outer_method) REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartiera1467d02017-02-22 09:22:50 -080035 // For AOT code, we need a write barrier for the class loader that holds the
36 // GC roots in the .bss.
37 const DexFile* dex_file = outer_method->GetDexFile();
Vladimir Marko6bec91c2017-01-09 15:03:12 +000038 if (dex_file != nullptr &&
39 dex_file->GetOatDexFile() != nullptr &&
40 !dex_file->GetOatDexFile()->GetOatFile()->GetBssGcRoots().empty()) {
Vladimir Marko9b03cb42017-02-16 16:37:03 +000041 ObjPtr<mirror::ClassLoader> class_loader = outer_method->GetClassLoader();
42 if (kIsDebugBuild) {
43 ClassTable* class_table =
44 Runtime::Current()->GetClassLinker()->ClassTableForClassLoader(class_loader);
45 CHECK(class_table != nullptr &&
46 !class_table->InsertOatFile(dex_file->GetOatDexFile()->GetOatFile()))
Vladimir Marko1998cd02017-01-13 13:02:58 +000047 << "Oat file with .bss GC roots was not registered in class table: "
48 << dex_file->GetOatDexFile()->GetOatFile()->GetLocation();
Vladimir Marko9b03cb42017-02-16 16:37:03 +000049 }
Mathieu Chartiera1467d02017-02-22 09:22:50 -080050 if (class_loader != nullptr) {
51 // Note that we emit the barrier before the compiled code stores the String or Class
52 // as a GC root. This is OK as there is no suspend point point in between.
53 Runtime::Current()->GetHeap()->WriteBarrierEveryFieldOf(class_loader);
54 } else {
55 Runtime::Current()->GetClassLinker()->WriteBarrierForBootOatFileBssRoots(
56 dex_file->GetOatDexFile()->GetOatFile());
Vladimir Marko1998cd02017-01-13 13:02:58 +000057 }
Vladimir Marko6bec91c2017-01-09 15:03:12 +000058 }
59}
60
Vladimir Markoea4c1262017-02-06 19:59:33 +000061constexpr Runtime::CalleeSaveType kInitEntrypointSaveType =
62 // TODO: Change allocation entrypoints on MIPS and MIPS64 to kSaveEverything.
63 (kRuntimeISA == kMips || kRuntimeISA == kMips64) ? Runtime::kSaveRefsOnly
64 : Runtime::kSaveEverything;
65
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +010066extern "C" mirror::Class* artInitializeStaticStorageFromCode(uint32_t type_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070067 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerse2645d32012-04-11 14:42:42 -070068 // Called to ensure static storage base is initialized for direct static field reads and writes.
69 // A class may be accessing another class' fields when it doesn't have access, as access has been
70 // given by inheritance.
Ian Rogers1d8cdbc2014-09-22 22:51:09 -070071 ScopedQuickEntrypointChecks sqec(self);
Vladimir Markoea4c1262017-02-06 19:59:33 +000072 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(self, kInitEntrypointSaveType);
Vladimir Marko6bec91c2017-01-09 15:03:12 +000073 ArtMethod* caller = caller_and_outer.caller;
74 mirror::Class* result =
75 ResolveVerifyAndClinit(dex::TypeIndex(type_idx), caller, self, true, false);
76 if (LIKELY(result != nullptr)) {
77 BssWriteBarrier(caller_and_outer.outer_method);
78 }
79 return result;
Ian Rogers57b86d42012-03-27 16:05:41 -070080}
81
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +010082extern "C" mirror::Class* artInitializeTypeFromCode(uint32_t type_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070083 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogerse2645d32012-04-11 14:42:42 -070084 // Called when method->dex_cache_resolved_types_[] misses.
Ian Rogers1d8cdbc2014-09-22 22:51:09 -070085 ScopedQuickEntrypointChecks sqec(self);
Vladimir Markoea4c1262017-02-06 19:59:33 +000086 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(self, kInitEntrypointSaveType);
Vladimir Marko6bec91c2017-01-09 15:03:12 +000087 ArtMethod* caller = caller_and_outer.caller;
88 mirror::Class* result =
89 ResolveVerifyAndClinit(dex::TypeIndex(type_idx), caller, self, false, false);
90 if (LIKELY(result != nullptr)) {
91 BssWriteBarrier(caller_and_outer.outer_method);
92 }
93 return result;
Ian Rogers57b86d42012-03-27 16:05:41 -070094}
95
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +010096extern "C" mirror::Class* artInitializeTypeAndVerifyAccessFromCode(uint32_t type_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070097 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogers57b86d42012-03-27 16:05:41 -070098 // 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 -070099 // unpopulated.
Ian Rogers1d8cdbc2014-09-22 22:51:09 -0700100 ScopedQuickEntrypointChecks sqec(self);
Vladimir Markoea4c1262017-02-06 19:59:33 +0000101 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(self, kInitEntrypointSaveType);
Vladimir Marko6bec91c2017-01-09 15:03:12 +0000102 ArtMethod* caller = caller_and_outer.caller;
103 mirror::Class* result =
104 ResolveVerifyAndClinit(dex::TypeIndex(type_idx), caller, self, false, true);
105 if (LIKELY(result != nullptr)) {
106 BssWriteBarrier(caller_and_outer.outer_method);
107 }
108 return result;
Ian Rogers57b86d42012-03-27 16:05:41 -0700109}
110
Nicolas Geoffray7ea6a172015-05-19 18:58:54 +0100111extern "C" mirror::String* artResolveStringFromCode(int32_t string_idx, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700112 REQUIRES_SHARED(Locks::mutator_lock_) {
Ian Rogers1d8cdbc2014-09-22 22:51:09 -0700113 ScopedQuickEntrypointChecks sqec(self);
Vladimir Markoea4c1262017-02-06 19:59:33 +0000114 auto caller_and_outer = GetCalleeSaveMethodCallerAndOuterMethod(self, kInitEntrypointSaveType);
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