blob: db214e7983d967b8d4b5e8b14fb22b3ba16363e4 [file] [log] [blame]
Mathieu Chartiere5f13e52015-02-24 09:37:21 -08001/*
2 * Copyright 2014 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#ifndef ART_RUNTIME_JIT_JIT_CODE_CACHE_H_
18#define ART_RUNTIME_JIT_JIT_CODE_CACHE_H_
19
20#include "instrumentation.h"
21
22#include "atomic.h"
Mingyao Yang063fc772016-08-02 11:02:54 -070023#include "base/arena_containers.h"
Nicolas Geoffray933330a2016-03-16 14:20:06 +000024#include "base/histogram-inl.h"
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080025#include "base/macros.h"
26#include "base/mutex.h"
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +010027#include "gc/accounting/bitmap.h"
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080028#include "gc_root.h"
29#include "jni.h"
Calin Juravle99629622016-04-19 16:33:46 +010030#include "method_reference.h"
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080031#include "oat_file.h"
32#include "object_callbacks.h"
Calin Juravle940eb0c2017-01-30 19:30:44 -080033#include "profile_compilation_info.h"
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080034#include "safe_map.h"
35#include "thread_pool.h"
36
37namespace art {
38
Mathieu Chartiere401d142015-04-22 13:56:20 -070039class ArtMethod;
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +010040class LinearAlloc;
Nicolas Geoffraye51ca8b2016-11-22 14:49:31 +000041class InlineCache;
Nicolas Geoffray26705e22015-10-28 12:50:11 +000042class ProfilingInfo;
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080043
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080044namespace jit {
45
46class JitInstrumentationCache;
47
Nicolas Geoffray0a3be162015-11-18 11:15:22 +000048// Alignment in bits that will suit all architectures.
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +010049static constexpr int kJitCodeAlignment = 16;
50using CodeCacheBitmap = gc::accounting::MemoryRangeBitmap<kJitCodeAlignment>;
51
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080052class JitCodeCache {
53 public:
Nicolas Geoffray0a3be162015-11-18 11:15:22 +000054 static constexpr size_t kMaxCapacity = 64 * MB;
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +010055 // Put the default to a very low amount for debug builds to stress the code cache
56 // collection.
Nicolas Geoffray7ca4b772016-02-23 13:52:01 +000057 static constexpr size_t kInitialCapacity = kIsDebugBuild ? 8 * KB : 64 * KB;
Nicolas Geoffray65b83d82016-02-22 13:14:04 +000058
59 // By default, do not GC until reaching 256KB.
60 static constexpr size_t kReservedCapacity = kInitialCapacity * 4;
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080061
Mathieu Chartierbce416f2015-03-23 12:37:35 -070062 // Create the code cache with a code + data capacity equal to "capacity", error message is passed
63 // in the out arg error_msg.
Nicolas Geoffraya25dce92016-01-12 16:41:10 +000064 static JitCodeCache* Create(size_t initial_capacity,
65 size_t max_capacity,
66 bool generate_debug_info,
67 std::string* error_msg);
Mathieu Chartiere5f13e52015-02-24 09:37:21 -080068
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +010069 // Number of bytes allocated in the code cache.
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +010070 size_t CodeCacheSize() REQUIRES(!lock_);
71
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +010072 // Number of bytes allocated in the data cache.
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +010073 size_t DataCacheSize() REQUIRES(!lock_);
74
Nicolas Geoffrayb331feb2016-02-05 16:51:53 +000075 bool NotifyCompilationOf(ArtMethod* method, Thread* self, bool osr)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070076 REQUIRES_SHARED(Locks::mutator_lock_)
Nicolas Geoffray73be1e82015-09-17 15:22:56 +010077 REQUIRES(!lock_);
78
Alex Lightdba61482016-12-21 08:20:29 -080079 void NotifyMethodRedefined(ArtMethod* method)
80 REQUIRES(Locks::mutator_lock_)
81 REQUIRES(!lock_);
82
Nicolas Geoffray07e3ca92016-03-11 09:57:57 +000083 // Notify to the code cache that the compiler wants to use the
84 // profiling info of `method` to drive optimizations,
85 // and therefore ensure the returned profiling info object is not
86 // collected.
87 ProfilingInfo* NotifyCompilerUse(ArtMethod* method, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070088 REQUIRES_SHARED(Locks::mutator_lock_)
Nicolas Geoffrayb6e20ae2016-03-07 14:29:04 +000089 REQUIRES(!lock_);
90
buzbee454b3b62016-04-07 14:42:47 -070091 void DoneCompiling(ArtMethod* method, Thread* self, bool osr)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070092 REQUIRES_SHARED(Locks::mutator_lock_)
Nicolas Geoffray73be1e82015-09-17 15:22:56 +010093 REQUIRES(!lock_);
94
Nicolas Geoffray07e3ca92016-03-11 09:57:57 +000095 void DoneCompilerUse(ArtMethod* method, Thread* self)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -070096 REQUIRES_SHARED(Locks::mutator_lock_)
Nicolas Geoffrayb6e20ae2016-03-07 14:29:04 +000097 REQUIRES(!lock_);
98
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +010099 // Allocate and write code and its metadata to the code cache.
Mingyao Yang063fc772016-08-02 11:02:54 -0700100 // `cha_single_implementation_list` needs to be registered via CHA (if it's
101 // still valid), since the compiled code still needs to be invalidated if the
102 // single-implementation assumptions are violated later. This needs to be done
103 // even if `has_should_deoptimize_flag` is false, which can happen due to CHA
104 // guard elimination.
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +0100105 uint8_t* CommitCode(Thread* self,
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100106 ArtMethod* method,
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000107 uint8_t* stack_map,
Mathieu Chartiercbcedbf2017-03-12 22:24:50 -0700108 uint8_t* method_info,
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000109 uint8_t* roots_data,
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +0100110 size_t frame_size_in_bytes,
111 size_t core_spill_mask,
112 size_t fp_spill_mask,
113 const uint8_t* code,
Nicolas Geoffrayb331feb2016-02-05 16:51:53 +0000114 size_t code_size,
Nicolas Geoffrayed015ac2016-12-15 17:58:48 +0000115 size_t data_size,
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000116 bool osr,
Mingyao Yang063fc772016-08-02 11:02:54 -0700117 Handle<mirror::ObjectArray<mirror::Object>> roots,
118 bool has_should_deoptimize_flag,
119 const ArenaSet<ArtMethod*>& cha_single_implementation_list)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700120 REQUIRES_SHARED(Locks::mutator_lock_)
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +0100121 REQUIRES(!lock_);
122
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100123 // Return true if the code cache contains this pc.
124 bool ContainsPc(const void* pc) const;
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800125
Nicolas Geoffraya5891e82015-11-06 14:18:27 +0000126 // Return true if the code cache contains this method.
127 bool ContainsMethod(ArtMethod* method) REQUIRES(!lock_);
128
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000129 // Allocate a region of data that contain `size` bytes, and potentially space
130 // for storing `number_of_roots` roots. Returns null if there is no more room.
Nicolas Geoffrayed015ac2016-12-15 17:58:48 +0000131 // Return the number of bytes allocated.
132 size_t ReserveData(Thread* self,
Mathieu Chartiercbcedbf2017-03-12 22:24:50 -0700133 size_t stack_map_size,
134 size_t method_info_size,
Nicolas Geoffrayed015ac2016-12-15 17:58:48 +0000135 size_t number_of_roots,
136 ArtMethod* method,
137 uint8_t** stack_map_data,
Mathieu Chartiercbcedbf2017-03-12 22:24:50 -0700138 uint8_t** method_info_data,
Nicolas Geoffrayed015ac2016-12-15 17:58:48 +0000139 uint8_t** roots_data)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700140 REQUIRES_SHARED(Locks::mutator_lock_)
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100141 REQUIRES(!lock_);
Nicolas Geoffray5550ca82015-08-21 18:38:30 +0100142
Nicolas Geoffrayd28b9692015-11-04 14:36:55 +0000143 // Clear data from the data portion of the code cache.
Nicolas Geoffrayf46501c2016-11-22 13:45:36 +0000144 void ClearData(Thread* self, uint8_t* stack_map_data, uint8_t* roots_data)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700145 REQUIRES_SHARED(Locks::mutator_lock_)
Nicolas Geoffrayd28b9692015-11-04 14:36:55 +0000146 REQUIRES(!lock_);
147
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100148 CodeCacheBitmap* GetLiveBitmap() const {
149 return live_bitmap_.get();
150 }
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800151
Nicolas Geoffray35122442016-03-02 12:05:30 +0000152 // Return whether we should do a full collection given the current state of the cache.
153 bool ShouldDoFullCollection()
154 REQUIRES(lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700155 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray35122442016-03-02 12:05:30 +0000156
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100157 // Perform a collection on the code cache.
158 void GarbageCollectCache(Thread* self)
159 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700160 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100161
162 // Given the 'pc', try to find the JIT compiled code associated with it.
163 // Return null if 'pc' is not in the code cache. 'method' is passed for
164 // sanity check.
165 OatQuickMethodHeader* LookupMethodHeader(uintptr_t pc, ArtMethod* method)
166 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700167 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100168
Nicolas Geoffrayb331feb2016-02-05 16:51:53 +0000169 OatQuickMethodHeader* LookupOsrMethodHeader(ArtMethod* method)
170 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700171 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffrayb331feb2016-02-05 16:51:53 +0000172
Nicolas Geoffray26705e22015-10-28 12:50:11 +0000173 // Remove all methods in our cache that were allocated by 'alloc'.
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100174 void RemoveMethodsIn(Thread* self, const LinearAlloc& alloc)
175 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700176 REQUIRES_SHARED(Locks::mutator_lock_);
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800177
Nicolas Geoffraye51ca8b2016-11-22 14:49:31 +0000178 void CopyInlineCacheInto(const InlineCache& ic, Handle<mirror::ObjectArray<mirror::Class>> array)
179 REQUIRES(!lock_)
180 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffrayb6e20ae2016-03-07 14:29:04 +0000181
Nicolas Geoffray26705e22015-10-28 12:50:11 +0000182 // Create a 'ProfileInfo' for 'method'. If 'retry_allocation' is true,
183 // will collect and retry if the first allocation is unsuccessful.
184 ProfilingInfo* AddProfilingInfo(Thread* self,
185 ArtMethod* method,
186 const std::vector<uint32_t>& entries,
187 bool retry_allocation)
188 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700189 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray26705e22015-10-28 12:50:11 +0000190
Nicolas Geoffray0a3be162015-11-18 11:15:22 +0000191 bool OwnsSpace(const void* mspace) const NO_THREAD_SAFETY_ANALYSIS {
192 return mspace == code_mspace_ || mspace == data_mspace_;
193 }
194
195 void* MoreCore(const void* mspace, intptr_t increment);
196
Calin Juravle99629622016-04-19 16:33:46 +0100197 // Adds to `methods` all profiled methods which are part of any of the given dex locations.
198 void GetProfiledMethods(const std::set<std::string>& dex_base_locations,
Calin Juravle940eb0c2017-01-30 19:30:44 -0800199 std::vector<ProfileMethodInfo>& methods)
Calin Juravle31f2c152015-10-23 17:56:15 +0100200 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700201 REQUIRES_SHARED(Locks::mutator_lock_);
Calin Juravle31f2c152015-10-23 17:56:15 +0100202
Calin Juravle4d77b6a2015-12-01 18:38:09 +0000203 uint64_t GetLastUpdateTimeNs() const;
Calin Juravle31f2c152015-10-23 17:56:15 +0100204
Nicolas Geoffrayaee21562015-12-15 16:39:44 +0000205 size_t GetCurrentCapacity() REQUIRES(!lock_) {
206 MutexLock lock(Thread::Current(), lock_);
207 return current_capacity_;
208 }
209
Nicolas Geoffraya25dce92016-01-12 16:41:10 +0000210 size_t GetMemorySizeOfCodePointer(const void* ptr) REQUIRES(!lock_);
211
Nicolas Geoffrayb88d59e2016-02-17 11:31:49 +0000212 void InvalidateCompiledCodeFor(ArtMethod* method, const OatQuickMethodHeader* code)
213 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700214 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffrayb88d59e2016-02-17 11:31:49 +0000215
Nicolas Geoffraybcd94c82016-03-03 13:23:33 +0000216 void Dump(std::ostream& os) REQUIRES(!lock_);
217
Nicolas Geoffray71cd50f2016-04-14 15:00:33 +0100218 bool IsOsrCompiled(ArtMethod* method) REQUIRES(!lock_);
219
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000220 void SweepRootTables(IsMarkedVisitor* visitor)
221 REQUIRES(!lock_)
222 REQUIRES_SHARED(Locks::mutator_lock_);
223
Nicolas Geoffraye51ca8b2016-11-22 14:49:31 +0000224 // The GC needs to disallow the reading of inline caches when it processes them,
225 // to avoid having a class being used while it is being deleted.
226 void AllowInlineCacheAccess() REQUIRES(!lock_);
227 void DisallowInlineCacheAccess() REQUIRES(!lock_);
228 void BroadcastForInlineCacheAccess() REQUIRES(!lock_);
229
Alex Lightdba61482016-12-21 08:20:29 -0800230 // Notify the code cache that the method at the pointer 'old_method' is being moved to the pointer
231 // 'new_method' since it is being made obsolete.
232 void MoveObsoleteMethod(ArtMethod* old_method, ArtMethod* new_method)
233 REQUIRES(!lock_) REQUIRES(Locks::mutator_lock_);
234
Nicolas Geoffray169722b2017-02-27 14:01:59 +0000235 // Dynamically change whether we want to garbage collect code. Should only be used
236 // by tests.
237 void SetGarbageCollectCode(bool value) {
238 garbage_collect_code_ = value;
239 }
240
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800241 private:
Nicolas Geoffray0a3be162015-11-18 11:15:22 +0000242 // Take ownership of maps.
243 JitCodeCache(MemMap* code_map,
244 MemMap* data_map,
245 size_t initial_code_capacity,
246 size_t initial_data_capacity,
Nicolas Geoffraya25dce92016-01-12 16:41:10 +0000247 size_t max_capacity,
248 bool garbage_collect_code);
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800249
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100250 // Internal version of 'CommitCode' that will not retry if the
251 // allocation fails. Return null if the allocation fails.
252 uint8_t* CommitCodeInternal(Thread* self,
253 ArtMethod* method,
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000254 uint8_t* stack_map,
Mathieu Chartiercbcedbf2017-03-12 22:24:50 -0700255 uint8_t* method_info,
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000256 uint8_t* roots_data,
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100257 size_t frame_size_in_bytes,
258 size_t core_spill_mask,
259 size_t fp_spill_mask,
260 const uint8_t* code,
Nicolas Geoffrayb331feb2016-02-05 16:51:53 +0000261 size_t code_size,
Nicolas Geoffrayed015ac2016-12-15 17:58:48 +0000262 size_t data_size,
Nicolas Geoffray132d8362016-11-16 09:19:42 +0000263 bool osr,
Mingyao Yang063fc772016-08-02 11:02:54 -0700264 Handle<mirror::ObjectArray<mirror::Object>> roots,
265 bool has_should_deoptimize_flag,
266 const ArenaSet<ArtMethod*>& cha_single_implementation_list)
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100267 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700268 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100269
Nicolas Geoffray26705e22015-10-28 12:50:11 +0000270 ProfilingInfo* AddProfilingInfoInternal(Thread* self,
271 ArtMethod* method,
272 const std::vector<uint32_t>& entries)
Nicolas Geoffray1e7da9b2016-03-01 14:11:40 +0000273 REQUIRES(lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700274 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray26705e22015-10-28 12:50:11 +0000275
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100276 // If a collection is in progress, wait for it to finish. Return
277 // whether the thread actually waited.
278 bool WaitForPotentialCollectionToComplete(Thread* self)
279 REQUIRES(lock_) REQUIRES(!Locks::mutator_lock_);
280
Mingyao Yang063fc772016-08-02 11:02:54 -0700281 // Remove CHA dependents and underlying allocations for entries in `method_headers`.
282 void FreeAllMethodHeaders(const std::unordered_set<OatQuickMethodHeader*>& method_headers)
283 REQUIRES(!lock_)
284 REQUIRES(!Locks::cha_lock_);
285
286 // Free in the mspace allocations for `code_ptr`.
287 void FreeCode(const void* code_ptr) REQUIRES(lock_);
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100288
Nicolas Geoffraya5891e82015-11-06 14:18:27 +0000289 // Number of bytes allocated in the code cache.
290 size_t CodeCacheSizeLocked() REQUIRES(lock_);
291
292 // Number of bytes allocated in the data cache.
293 size_t DataCacheSizeLocked() REQUIRES(lock_);
294
Nicolas Geoffray0a3be162015-11-18 11:15:22 +0000295 // Notify all waiting threads that a collection is done.
296 void NotifyCollectionDone(Thread* self) REQUIRES(lock_);
297
298 // Try to increase the current capacity of the code cache. Return whether we
299 // succeeded at doing so.
300 bool IncreaseCodeCacheCapacity() REQUIRES(lock_);
301
302 // Set the footprint limit of the code cache.
303 void SetFootprintLimit(size_t new_footprint) REQUIRES(lock_);
304
Nicolas Geoffray35122442016-03-02 12:05:30 +0000305 void DoCollection(Thread* self, bool collect_profiling_info)
Nicolas Geoffray8d372502016-02-23 13:56:43 +0000306 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700307 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray8d372502016-02-23 13:56:43 +0000308
Nicolas Geoffray9abb2972016-03-04 14:32:59 +0000309 void RemoveUnmarkedCode(Thread* self)
Nicolas Geoffray8d372502016-02-23 13:56:43 +0000310 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700311 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray8d372502016-02-23 13:56:43 +0000312
313 void MarkCompiledCodeOnThreadStacks(Thread* self)
314 REQUIRES(!lock_)
Andreas Gampebdf7f1c2016-08-30 16:38:47 -0700315 REQUIRES_SHARED(Locks::mutator_lock_);
Nicolas Geoffray8d372502016-02-23 13:56:43 +0000316
Nicolas Geoffray35122442016-03-02 12:05:30 +0000317 bool CheckLiveCompiledCodeHasProfilingInfo()
Nicolas Geoffraycf48fa02016-07-30 22:49:11 +0100318 REQUIRES(lock_);
Nicolas Geoffray35122442016-03-02 12:05:30 +0000319
Nicolas Geoffraybcd94c82016-03-03 13:23:33 +0000320 void FreeCode(uint8_t* code) REQUIRES(lock_);
321 uint8_t* AllocateCode(size_t code_size) REQUIRES(lock_);
322 void FreeData(uint8_t* data) REQUIRES(lock_);
323 uint8_t* AllocateData(size_t data_size) REQUIRES(lock_);
324
Nicolas Geoffraye51ca8b2016-11-22 14:49:31 +0000325 bool IsWeakAccessEnabled(Thread* self) const;
326 void WaitUntilInlineCacheAccessible(Thread* self)
327 REQUIRES(!lock_)
328 REQUIRES_SHARED(Locks::mutator_lock_);
329
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100330 // Lock for guarding allocations, collections, and the method_code_map_.
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800331 Mutex lock_;
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100332 // Condition to wait on during collection.
333 ConditionVariable lock_cond_ GUARDED_BY(lock_);
334 // Whether there is a code cache collection in progress.
335 bool collection_in_progress_ GUARDED_BY(lock_);
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +0100336 // Mem map which holds code.
337 std::unique_ptr<MemMap> code_map_;
338 // Mem map which holds data (stack maps and profiling info).
339 std::unique_ptr<MemMap> data_map_;
340 // The opaque mspace for allocating code.
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100341 void* code_mspace_ GUARDED_BY(lock_);
Nicolas Geoffray0c3c2662015-10-15 13:53:04 +0100342 // The opaque mspace for allocating data.
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100343 void* data_mspace_ GUARDED_BY(lock_);
344 // Bitmap for collecting code and data.
345 std::unique_ptr<CodeCacheBitmap> live_bitmap_;
Nicolas Geoffrayb331feb2016-02-05 16:51:53 +0000346 // Holds compiled code associated to the ArtMethod.
Nicolas Geoffray1dad3f62015-10-23 14:59:54 +0100347 SafeMap<const void*, ArtMethod*> method_code_map_ GUARDED_BY(lock_);
Nicolas Geoffrayb331feb2016-02-05 16:51:53 +0000348 // Holds osr compiled code associated to the ArtMethod.
349 SafeMap<ArtMethod*, const void*> osr_code_map_ GUARDED_BY(lock_);
Nicolas Geoffray26705e22015-10-28 12:50:11 +0000350 // ProfilingInfo objects we have allocated.
351 std::vector<ProfilingInfo*> profiling_infos_ GUARDED_BY(lock_);
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800352
Nicolas Geoffray0a3be162015-11-18 11:15:22 +0000353 // The maximum capacity in bytes this code cache can go to.
354 size_t max_capacity_ GUARDED_BY(lock_);
355
356 // The current capacity in bytes of the code cache.
357 size_t current_capacity_ GUARDED_BY(lock_);
358
359 // The current footprint in bytes of the code portion of the code cache.
360 size_t code_end_ GUARDED_BY(lock_);
361
362 // The current footprint in bytes of the data portion of the code cache.
363 size_t data_end_ GUARDED_BY(lock_);
364
Nicolas Geoffray35122442016-03-02 12:05:30 +0000365 // Whether the last collection round increased the code cache.
366 bool last_collection_increased_code_cache_ GUARDED_BY(lock_);
Nicolas Geoffray0a3be162015-11-18 11:15:22 +0000367
Calin Juravle31f2c152015-10-23 17:56:15 +0100368 // Last time the the code_cache was updated.
Calin Juravle4d77b6a2015-12-01 18:38:09 +0000369 // It is atomic to avoid locking when reading it.
370 Atomic<uint64_t> last_update_time_ns_;
Calin Juravle31f2c152015-10-23 17:56:15 +0100371
Nicolas Geoffray169722b2017-02-27 14:01:59 +0000372 // Whether we can do garbage collection. Not 'const' as tests may override this.
373 bool garbage_collect_code_;
Nicolas Geoffraya25dce92016-01-12 16:41:10 +0000374
Nicolas Geoffray38ea9bd2016-02-19 16:25:57 +0000375 // The size in bytes of used memory for the data portion of the code cache.
376 size_t used_memory_for_data_ GUARDED_BY(lock_);
377
378 // The size in bytes of used memory for the code portion of the code cache.
379 size_t used_memory_for_code_ GUARDED_BY(lock_);
380
Nicolas Geoffray0a522232016-01-19 09:34:58 +0000381 // Number of compilations done throughout the lifetime of the JIT.
382 size_t number_of_compilations_ GUARDED_BY(lock_);
Nicolas Geoffraybcd94c82016-03-03 13:23:33 +0000383
384 // Number of compilations for on-stack-replacement done throughout the lifetime of the JIT.
Nicolas Geoffrayfcdd7292016-02-25 13:27:47 +0000385 size_t number_of_osr_compilations_ GUARDED_BY(lock_);
Nicolas Geoffray0a522232016-01-19 09:34:58 +0000386
Nicolas Geoffraybcd94c82016-03-03 13:23:33 +0000387 // Number of deoptimizations done throughout the lifetime of the JIT.
388 size_t number_of_deoptimizations_ GUARDED_BY(lock_);
389
390 // Number of code cache collections done throughout the lifetime of the JIT.
391 size_t number_of_collections_ GUARDED_BY(lock_);
392
Nicolas Geoffray933330a2016-03-16 14:20:06 +0000393 // Histograms for keeping track of stack map size statistics.
394 Histogram<uint64_t> histogram_stack_map_memory_use_ GUARDED_BY(lock_);
395
396 // Histograms for keeping track of code size statistics.
397 Histogram<uint64_t> histogram_code_memory_use_ GUARDED_BY(lock_);
398
399 // Histograms for keeping track of profiling info statistics.
400 Histogram<uint64_t> histogram_profiling_info_memory_use_ GUARDED_BY(lock_);
401
Nicolas Geoffraye51ca8b2016-11-22 14:49:31 +0000402 // Whether the GC allows accessing weaks in inline caches. Note that this
403 // is not used by the concurrent collector, which uses
404 // Thread::SetWeakRefAccessEnabled instead.
405 Atomic<bool> is_weak_access_enabled_;
406
407 // Condition to wait on for accessing inline caches.
408 ConditionVariable inline_cache_cond_ GUARDED_BY(lock_);
409
Mathieu Chartier3130cdf2015-05-03 15:20:23 -0700410 DISALLOW_IMPLICIT_CONSTRUCTORS(JitCodeCache);
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800411};
412
Mathieu Chartiere5f13e52015-02-24 09:37:21 -0800413} // namespace jit
414} // namespace art
415
416#endif // ART_RUNTIME_JIT_JIT_CODE_CACHE_H_