-
Notifications
You must be signed in to change notification settings - Fork 139
/
Copy pathOMRCodeCacheManager.hpp
394 lines (332 loc) · 16.8 KB
/
OMRCodeCacheManager.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
/*******************************************************************************
* Copyright IBM Corp. and others 2000
*
* This program and the accompanying materials are made available under
* the terms of the Eclipse Public License 2.0 which accompanies this
* distribution and is available at https://www.eclipse.org/legal/epl-2.0/
* or the Apache License, Version 2.0 which accompanies this distribution
* and is available at https://www.apache.org/licenses/LICENSE-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License, v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception [1] and GNU General Public
* License, version 2 with the OpenJDK Assembly Exception [2].
*
* [1] https://www.gnu.org/software/classpath/license.html
* [2] https://openjdk.org/legal/assembly-exception.html
*
* SPDX-License-Identifier: EPL-2.0 OR Apache-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0 OR GPL-2.0-only WITH OpenJDK-assembly-exception-1.0
*******************************************************************************/
#ifndef OMR_CODECACHEMANAGER_INCL
#define OMR_CODECACHEMANAGER_INCL
#include <stddef.h>
#include <stdint.h>
#include "il/DataTypes.hpp"
#include "infra/CriticalSection.hpp"
#include "runtime/CodeCacheConfig.hpp"
#include "runtime/MethodExceptionData.hpp"
#include "runtime/Runtime.hpp"
#include "runtime/CodeCacheTypes.hpp"
#include "env/RawAllocator.hpp"
#include "codegen/StaticRelocation.hpp"
#include "codegen/ELFRelocationResolver.hpp"
class TR_FrontEnd;
class TR_OpaqueMethodBlock;
class TR_Memory;
namespace TR { class CodeCache; }
namespace TR { class CodeCacheManager; }
namespace TR { class CodeCacheMemorySegment; }
namespace TR { class CodeGenerator; }
namespace TR { class Monitor; }
namespace OMR { class CodeCacheHashEntrySlab; }
namespace OMR { typedef void CodeCacheTrampolineCode; }
namespace OMR { class CodeCacheManager; }
namespace TR { class StaticRelocation; }
namespace OMR { typedef CodeCacheManager CodeCacheManagerConnector; }
#if (HOST_OS == OMR_LINUX)
namespace TR { class ELFRelocatableGenerator; }
namespace TR { class ELFExecutableGenerator; }
namespace TR {
/**
* Structure used to track regions of code cache that will become symbols
*/
typedef struct CodeCacheSymbol{
const char *_name; /**< Symbol name */
uint32_t _nameLength; /**< start of symbol name */
uint8_t *_start; /**< Start PC */
uint32_t _size; /**< Code size */
struct CodeCacheSymbol *_next; /**< ptr to next CodeCacheSymbol */
} CodeCacheSymbol;
/**
* Structure used to track the CodeCacheSymbol structs connected through a
* linked list data structure.
*/
typedef struct CodeCacheSymbolContainer{
CodeCacheSymbol *_head; /**< ptr to the first CodeCacheSymbol */
CodeCacheSymbol *_tail; /**< ptr to the latest CodeCacheSymbol */
uint32_t _numSymbols; /**< Number of symbols in the linked list */
uint32_t _totalSymbolNameLength; /**< Sum of the symbol names in the linked list */
} CodeCacheSymbolContainer;
/**
* Structure used for tracking relocations
*/
typedef struct CodeCacheRelocationInfo{
uint8_t *_location; /**< the address requiring relocation */
uint32_t _type; /**< relocation type: absolute or relative */
uint32_t _symbol; /**< index of the symbol to be relocated in the linked list of CodeCacheSymbol, with head symbol being 0 */
struct CodeCacheRelocationInfo *_next; /**< link to the next symbol */
} CodeCacheRelocationInfo;
typedef struct CodeCacheRelocationInfoContainer{
CodeCacheRelocationInfo *_head; /**< First CodeCacheRelocationInfo structure */
CodeCacheRelocationInfo *_tail; /**< Latest CodeCacheRelocationInfo structure added */
uint32_t _numRelocations; /**< Total number of CodeCacheRelocationInfor in the linked list */
} CodeCacheRelocationInfoContainer;
} // namespace TR
#endif // HOST_OS == OMR_LINUX
namespace OMR {
class OMR_EXTENSIBLE CodeCacheManager
{
TR::CodeCacheManager *self();
protected:
struct CodeCacheList
{
TR::CodeCache *_head;
TR::Monitor *_mutex;
};
public:
CodeCacheManager(TR::RawAllocator rawAllocator);
class CacheListCriticalSection : public CriticalSection
{
public:
CacheListCriticalSection(TR::CodeCacheManager *mgr);
};
class RepositoryMonitorCriticalSection : public CriticalSection
{
public:
RepositoryMonitorCriticalSection(TR::CodeCacheManager *mgr);
};
class UsageMonitorCriticalSection : public CriticalSection
{
public:
UsageMonitorCriticalSection(TR::CodeCacheManager *mgr);
};
TR::CodeCacheConfig & codeCacheConfig() { return _config; }
/**
* @brief Inquires whether the code cache full flag has been set
*
* @return true if code cache full flag is set; false otherwise.
*/
bool codeCacheFull() { return _codeCacheFull; }
/**
* @brief Sets the flag indicating a full code cache
*/
void setCodeCacheFull() { _codeCacheFull = true; };
TR::CodeCache *initialize(bool useConsolidatedCache, uint32_t numberOfCodeCachesToCreateAtStartup);
void destroy();
// These two functions are for allocating Native backing memory for code cache structures
// In future these facilities should probably be implemented via cs2 allocators
void *getMemory(size_t sizeInBytes);
void freeMemory(void *memoryToFree);
TR::CodeCache * allocateCodeCacheObject(TR::CodeCacheMemorySegment *codeCacheSegment,
size_t codeCacheSize);
void * chooseCacheStartAddress(size_t repositorySize);
TR::CodeCache * getFirstCodeCache() { return _codeCacheList._head; }
int32_t getCurrentNumberOfCodeCaches() { return _curNumberOfCodeCaches; }
TR::Monitor * cacheListMutex() const { return _codeCacheList._mutex; }
void addCodeCache(TR::CodeCache *codeCache);
TR::CodeCacheMemorySegment *getNewCodeCacheMemorySegment(size_t segmentSize, size_t & codeCacheSizeAllocated);
void unreserveCodeCache(TR::CodeCache *codeCache);
TR::CodeCache * reserveCodeCache(bool compilationCodeAllocationsMustBeContiguous,
size_t sizeEstimate,
int32_t compThreadID,
int32_t *numReserved);
TR::CodeCache * getNewCodeCache(int32_t reservingCompThreadID);
uint8_t * allocateCodeMemory(size_t warmCodeSize,
size_t coldCodeSize,
TR::CodeCache **codeCache_pp,
uint8_t ** coldCode,
bool needsToBeContiguous,
bool isMethodHeaderNeeded=true);
/**
* @brief Allocate and initialize a new code cache from a new memory segment
*
* @param[in] segmentSizeInBytes : segment size to create the code cache from
* @param[in] reservingCompilationTID : thread id of the requestor
* if (reservingCompilationTID)
* <= -2 : no reservation is requested
* == -1 : the application thread is requesting the allocation; new code cache will be reserved
* >= 0 : a compilation thread is requesting the allocation; new code cache will be reserved
*
* @return if allocation is successful, the allocated TR::CodeCache object from
* the new segment; NULL otherwise.
*/
TR::CodeCache * allocateCodeCacheFromNewSegment(
size_t segmentSizeInBytes,
int32_t reservingCompilationTID);
TR::CodeCache * findCodeCacheFromPC(void *inCacheAddress);
/**
* @brief Inquires whether the given code address is in RX code.
*
* @details
* This function is suitable for calling either at runtime (outside of a
* compilation context) or during compilation of a method. This function
* is static to facilitate calling from diverse contexts.
*
* @param[in] address The code address inquired about.
* @param[in] jitConfig The jitConfig structure for this language environment
*
* @returns \c true if the code address is in RX code; \c false otherwise.
*/
static bool isAddressInRXCode(intptr_t address, void *jitConfig);
/**
* @brief Inquires whether the given method startPC code address is in RX code.
*
* @details
* This function allows implementing language runtimes to provide a more
* efficient version of this query if it is known that the provided address
* is the startPC of a method.
*
* This function is suitable for calling either at runtime (outside of a
* compilation context) or during compilation of a method.
*
* If the address provided via \a startPC is not a method startPC then the
* result of this function is undefined. Determining whether the given
* address is actually a startPC will defeat any optimization benefit from
* assuming it is.
*
* This function is static to facilitate calling from diverse contexts.
*
* @param[in] startPC The startPC of a method
* @param[in] jitConfig The jitConfig structure for this language environment
*
* @returns \c true if the code address is in RX code; \c false otherwise.
*/
static bool isStartPCInRXCode(intptr_t startPC, void *jitConfig);
/**
* @brief Finds a helper trampoline for the given helper reachable from the
* given code cache address.
*
* @param[in] helperIndex : the index of the helper requested
* @param[in] callSite : the call site address
*
* @return The address of the helper trampoline in the same code cache as
* the call site address
*/
intptr_t findHelperTrampoline(int32_t helperIndex, void *callSite);
CodeCacheTrampolineCode * findMethodTrampoline(TR_OpaqueMethodBlock *method, void *callingPC);
void synchronizeTrampolines();
CodeCacheTrampolineCode * replaceTrampoline(TR_OpaqueMethodBlock *method,
void *callSite,
void *oldTrampoline,
void *oldTargetPC,
void *newTargetPC,
bool needSync);
void performSizeAdjustments(size_t &warmCodeSize,
size_t &coldCodeSize,
bool needsToBeContiguous,
bool isMethodHeaderNeeded);
bool canAddNewCodeCache();
/**
* @brief Answers whether there is enough physical memory available to perform
* code cache allocation for the requested size.
*
* Downstream projects can provide an implementation for this method to check
* for safe memory limits before code cache allocation.
*
* @param requestedCodeCacheSize: requested code cache size
*
* @return true if there is enough physical memory; false otherwise
*/
bool isSufficientPhysicalMemoryAvailableForAllocation(size_t requestedCodeCacheSize);
// Code Cache Consolidation
TR::CodeCache *allocateRepositoryCodeCache();
TR::CodeCacheMemorySegment *allocateCodeCacheRepository(size_t repositorySize);
TR::CodeCacheMemorySegment *carveCodeCacheSpaceFromRepository(size_t segmentSize,
size_t &codeCacheSizeToAllocate);
void undoCarvingFromRepository(TR::CodeCacheMemorySegment *segment);
TR::CodeCacheMemorySegment *setupMemorySegmentFromRepository(uint8_t *start,
uint8_t *end,
size_t & codeCacheSizeToAllocate);
void freeMemorySegment(TR::CodeCacheMemorySegment *segment);
// sneaky accounting to provide the right external perception of how much space is used
void increaseFreeSpaceInCodeCacheRepository(size_t size);
void decreaseFreeSpaceInCodeCacheRepository(size_t size);
bool usingRepository() { return _codeCacheRepositorySegment != NULL; }
TR::CodeCache * getRepositoryCodeCacheAddress() { return _repositoryCodeCache; }
TR::Monitor * getCodeCacheRepositoryMonitor() { return _codeCacheRepositoryMonitor; }
uint8_t * allocateCodeMemoryWithRetries(size_t warmCodeSize,
size_t coldCodeSize,
TR::CodeCache **codeCache_pp,
int32_t allocationRetries,
uint8_t ** coldCode,
bool needsToBeContiguous,
bool isMethodHeaderNeeded=true);
void setHasFailedCodeCacheAllocation() { }
bool initialized() const { return _initialized; }
bool lowCodeCacheSpaceThresholdReached() { return _lowCodeCacheSpaceThresholdReached; }
void repositoryCodeCacheCreated();
void registerCompiledMethod(const char *sig, uint8_t *startPC, uint32_t codeSize);
void registerStaticRelocation(const TR::StaticRelocation &relocation);
/**
* @brief Hint to free a given code cache segment.
*
* This function is intended to allow code cache segments allocated by
* `allocateCodeCacheSegment()` to be freed. Because it may not always
* be safe to actually free code cache memory, callers should treat
* this function as a hint. It is up to downstream projects to decide
* how to handle these requests. By default, if no overrides are
* provided, no memory is ever freed.
*
* @param memSegment is a pointer to the TR::CodeCacheMemorySegment instance
* that handles the segment memory to be freed.
*/
void freeCodeCacheSegment(TR::CodeCacheMemorySegment * memSegment) {}
void increaseCurrTotalUsedInBytes(size_t size);
void decreaseCurrTotalUsedInBytes(size_t size);
size_t getCurrTotalUsedInBytes() const { return _currTotalUsedInBytes; }
size_t getMaxUsedInBytes() const { return _maxUsedInBytes; }
protected:
TR::RawAllocator _rawAllocator;
TR::CodeCacheConfig _config;
TR::CodeCache *_lastCache; /*!< last code cache round robined through */
CodeCacheList _codeCacheList; /*!< list of allocated code caches */
int32_t _curNumberOfCodeCaches;
// The following 3 fields are for implementation of code cache consolidation
TR::CodeCache *_repositoryCodeCache;
TR::CodeCacheMemorySegment *_codeCacheRepositorySegment;
TR::Monitor *_codeCacheRepositoryMonitor;
bool _initialized; /*!< flag to indicate if code cache manager has been initialized or not */
bool _lowCodeCacheSpaceThresholdReached; /*!< true if close to exhausting available code cache */
bool _codeCacheFull;
TR::Monitor *_usageMonitor;
size_t _currTotalUsedInBytes;
size_t _maxUsedInBytes;
#if (HOST_OS == OMR_LINUX)
public:
/**
* Initializes the Relocatable ELF Generator, if the option is enabled
* @param
* @return
*/
void initializeRelocatableELFGenerator(void);
/**
* Initializes the Executable ELFGenerator if the option is enabled
* @param
* @return
*/
void initializeExecutableELFGenerator(void);
protected:
TR::ELFExecutableGenerator *_elfExecutableGenerator; /**< Executable ELF generator */
TR::ELFRelocatableGenerator *_elfRelocatableGenerator; /**< Relocatable ELF generator */
// collect information on code cache symbols here, will be post processed into the elf trailer structure
static TR::CodeCacheSymbolContainer *_symbolContainer; /**< Symbol container used for tracking CodeCacheSymbols.
Note: This static member is not thread-safe. Synchronization is needed if multiple compilation threads are active */
TR::CodeCacheSymbolContainer *_relocatableSymbolContainer; /**< Symbol container used for tracking CodeCacheSymbols, for the purpose of writing to relocatable ELF object file */
TR::CodeCacheRelocationInfoContainer *_relocations; /**< for tracking relocation info */
TR::ELFRelocationResolver _resolver; /**< this translates between a TR::StaticRelocation and the ELF relocation type required for the platform */
const char *_objectFileName; /**< filename of the object file to generate, obtained from cmd line options */
#endif // HOST_OS == OMR_LINUX
};
} // namespace OMR
#endif // OMR_CODECACHEMANAGER_INCL