From 688e7869d18be350f3e32b6a6bd8f347d722b3dd Mon Sep 17 00:00:00 2001 From: Pat Gavlin Date: Fri, 26 Feb 2016 11:17:03 -0800 Subject: [PATCH] Refactor the pooled ArenaAllocator. The mark/release functionality on `ArenaAllocator` was only being used by the pooled allocator, and caused some complications in API and implementation. Instead of providing this functionality on all allocators, refactor this functionality out of `ArenaAllocator` and into a new subclass, `PooledAllocator`, that provides the singleton pooled allocator. A number of additional usability/reliability changes have been enabled as part of this change: - `ArenaAllocator::initialize` has been replaced with move assignment - Asserts have been added on all relevant instance methods to ensure that `ArenaAllocator` values are initialized before use - `ArenaAllocator::returnPooledAllocator` has been replaced by making `ArenaAllocator::destroy` virtual and having `PooledAllocator::destroy` return the singleton - The teardown of the pooled allocator is now thread-safe w.r.t. the shutdown of the arena allocator subsystem --- src/jit/alloc.cpp | 406 +++++++++++++++++++++++-------------------- src/jit/alloc.h | 44 +++-- src/jit/compiler.cpp | 23 +-- 3 files changed, 248 insertions(+), 225 deletions(-) diff --git a/src/jit/alloc.cpp b/src/jit/alloc.cpp index 6594bb91e95d..2b4177994a20 100644 --- a/src/jit/alloc.cpp +++ b/src/jit/alloc.cpp @@ -8,10 +8,38 @@ #pragma hdrstop #endif // defined(_MSC_VER) +//------------------------------------------------------------------------ +// PooledAllocator: +// This subclass of `ArenaAllocator` is a singleton that always keeps +// a single default-sized page allocated. We try to use the singleton +// allocator as often as possible (i.e. for all non-concurrent +// method compilations). +class PooledAllocator : public ArenaAllocator +{ +private: + enum + { + POOLED_ALLOCATOR_NOTINITIALIZED = 0, + POOLED_ALLOCATOR_IN_USE = 1, + POOLED_ALLOCATOR_AVAILABLE = 2, + POOLED_ALLOCATOR_SHUTDOWN = 3, + }; + + static PooledAllocator s_pooledAllocator; + static LONG s_pooledAllocatorState; + + PooledAllocator() : ArenaAllocator() {} + PooledAllocator(IEEMemoryManager* memoryManager); + +public: + void destroy() override; + + static void shutdown(); + + static ArenaAllocator* getPooledAllocator(IEEMemoryManager* memoryManager); +}; + size_t ArenaAllocator::s_defaultPageSize = 0; -ArenaAllocator* ArenaAllocator::s_pooledAllocator; -ArenaAllocator::MarkDescriptor ArenaAllocator::s_pooledAllocatorMark; -LONG ArenaAllocator::s_isPooledAllocatorInUse = 0; //------------------------------------------------------------------------ // ArenaAllocator::bypassHostAllocator: @@ -46,45 +74,60 @@ size_t ArenaAllocator::getDefaultPageSize() } //------------------------------------------------------------------------ -// ArenaAllocator::initialize: -// Intializes an arena allocator. +// ArenaAllocator::ArenaAllocator: +// Default-constructs an arena allocator. +ArenaAllocator::ArenaAllocator() + : m_memoryManager(nullptr) + , m_firstPage(nullptr) + , m_lastPage(nullptr) + , m_nextFreeByte(nullptr) + , m_lastFreeByte(nullptr) +{ +} + +//------------------------------------------------------------------------ +// ArenaAllocator::ArenaAllocator: +// Constructs an arena allocator. // // Arguments: // memoryManager - The `IEEMemoryManager` instance that will be used to // allocate memory for arena pages. -// -// shuoldPreallocate - True if the allocator should allocate an initial -// arena page as part of initialization. -// -// Return Value: -// True if initialization succeeded; false otherwise. -bool ArenaAllocator::initialize(IEEMemoryManager* memoryManager, bool shouldPreallocate) +ArenaAllocator::ArenaAllocator(IEEMemoryManager* memoryManager) + : m_memoryManager(memoryManager) + , m_firstPage(nullptr) + , m_lastPage(nullptr) + , m_nextFreeByte(nullptr) + , m_lastFreeByte(nullptr) { - assert(s_defaultPageSize != 0); + assert(getDefaultPageSize() != 0); + assert(isInitialized()); +} - m_memoryManager = memoryManager; +//------------------------------------------------------------------------ +// ArenaAllocator::operator=: +// Move-assigns a `ArenaAllocator`. +ArenaAllocator& ArenaAllocator::operator=(ArenaAllocator&& other) +{ + assert(!isInitialized()); - m_firstPage = nullptr; - m_lastPage = nullptr; - m_nextFreeByte = nullptr; - m_lastFreeByte = nullptr; + m_memoryManager = other.m_memoryManager; + m_firstPage = other.m_firstPage; + m_lastPage = other.m_lastPage; + m_nextFreeByte = other.m_nextFreeByte; + m_lastFreeByte = other.m_lastFreeByte; - bool result = true; - if (shouldPreallocate) - { - // Grab the initial page. - setErrorTrap(NULL, ArenaAllocator*, thisPtr, this) // ERROR TRAP: Start normal block - { - thisPtr->allocateNewPage(0); - } - impJitErrorTrap() // ERROR TRAP: The following block handles errors - { - result = false; - } - endErrorTrap() // ERROR TRAP: End - } + other.m_memoryManager = nullptr; + other.m_firstPage = nullptr; + other.m_lastPage = nullptr; + other.m_nextFreeByte = nullptr; + other.m_lastFreeByte = nullptr; + + return *this; +} - return result; +bool ArenaAllocator::isInitialized() +{ + return m_memoryManager != nullptr; } //------------------------------------------------------------------------ @@ -97,14 +140,21 @@ bool ArenaAllocator::initialize(IEEMemoryManager* memoryManager, bool shouldPrea // // Return Value: // A pointer to the first usable byte of the newly allocated page. -void* ArenaAllocator::allocateNewPage(size_t size) +void* ArenaAllocator::allocateNewPage(size_t size, bool canThrow) { + assert(isInitialized()); + size_t pageSize = sizeof(PageDescriptor) + size; // Check for integer overflow if (pageSize < size) { - NOMEM(); + if (canThrow) + { + NOMEM(); + } + + return nullptr; } // If the current page is now full, update a few statistics @@ -133,7 +183,12 @@ void* ArenaAllocator::allocateNewPage(size_t size) PageDescriptor* newPage = (PageDescriptor*)allocateHostMemory(pageSize); if (newPage == nullptr) { - NOMEM(); + if (canThrow) + { + NOMEM(); + } + + return nullptr; } // Append the new page to the end of the list @@ -166,14 +221,10 @@ void* ArenaAllocator::allocateNewPage(size_t size) //------------------------------------------------------------------------ // ArenaAllocator::destroy: // Performs any necessary teardown for an `ArenaAllocator`. -// -// Notes: -// This method walks from `m_firstPage` forward and releases the pages. -// Be careful no other thread has called `reset` at the same time. -// Otherwise, the page specified by `page` could be double-freed -// (VSW 600919). void ArenaAllocator::destroy() { + assert(isInitialized()); + // Free all of the allocated pages for (PageDescriptor* page = m_firstPage, *next; page != nullptr; page = next) { @@ -182,81 +233,13 @@ void ArenaAllocator::destroy() } // Clear out the allocator's fields + m_memoryManager = nullptr; m_firstPage = nullptr; m_lastPage = nullptr; m_nextFreeByte = nullptr; m_lastFreeByte = nullptr; } -//------------------------------------------------------------------------ -// ArenaAllocator::mark: -// Stores the current position of an `ArenaAllocator` in the given mark. -// -// Arguments: -// mark - The mark that will store the current position of the -// allocator. -void ArenaAllocator::mark(MarkDescriptor& mark) -{ - mark.m_page = m_lastPage; - mark.m_next = m_nextFreeByte; - mark.m_last = m_lastFreeByte; -} - -//------------------------------------------------------------------------ -// ArenaAllocator::reset: -// Resets the current position of an `ArenaAllocator` to the given -// mark, freeing any unused pages. -// -// Arguments: -// mark - The mark that stores the desired position for the allocator. -// -// Notes: -// This method may walk the page list backward and release the pages. -// Be careful no other thread is doing `destroy` as the same time. -// Otherwise, the page specified by `temp` could be double-freed -// (VSW 600919). -void ArenaAllocator::reset(MarkDescriptor& mark) -{ - // If the active page hasn't changed, just reset the position into the - // page and return. - if (m_lastPage == mark.m_page) - { - m_nextFreeByte = mark.m_next; - m_lastFreeByte = mark.m_last; - return; - } - - // Otherwise, free any new pages that were added. - void* last = mark.m_page; - - if (last == nullptr) - { - if (m_firstPage == nullptr) - { - return; - } - - m_nextFreeByte = m_firstPage->m_contents; - m_lastFreeByte = m_firstPage->m_pageBytes + (BYTE*)m_firstPage; - return; - } - - while (m_lastPage != last) - { - // Remove the last page from the end of the list - PageDescriptor* temp = m_lastPage; - m_lastPage = temp->m_previous; - - // The new last page has no next page - m_lastPage->m_next = nullptr; - - freeHostMemory(temp); - } - - m_nextFreeByte = mark.m_next; - m_lastFreeByte = mark.m_last; -} - // The debug version of the allocator may allocate directly from the // OS rather than going through the hosting APIs. In order to do so, // it must undef the macros that are usually in place to prevent @@ -279,6 +262,8 @@ void ArenaAllocator::reset(MarkDescriptor& mark) // A pointer to the allocated memory. void* ArenaAllocator::allocateHostMemory(size_t size) { + assert(isInitialized()); + #if defined(DEBUG) if (bypassHostAllocator()) { @@ -301,6 +286,8 @@ void* ArenaAllocator::allocateHostMemory(size_t size) // block - A pointer to the memory to free. void ArenaAllocator::freeHostMemory(void* block) { + assert(isInitialized()); + #if defined(DEBUG) if (bypassHostAllocator()) { @@ -335,6 +322,7 @@ void ArenaAllocator::freeHostMemory(void* block) // use-before-init problems. void* ArenaAllocator::allocateMemory(size_t size) { + assert(isInitialized()); assert(size != 0 && (size & (sizeof(int) - 1)) == 0); // Ensure that we always allocate in pointer sized increments. @@ -361,7 +349,7 @@ void* ArenaAllocator::allocateMemory(size_t size) if (m_nextFreeByte > m_lastFreeByte) { - block = allocateNewPage(size); + block = allocateNewPage(size, true); } memset(block, UninitializedWord(), size); @@ -378,6 +366,8 @@ void* ArenaAllocator::allocateMemory(size_t size) // See above. size_t ArenaAllocator::getTotalBytesAllocated() { + assert(isInitialized()); + size_t bytes = 0; for (PageDescriptor* page = m_firstPage; page != nullptr; page = page->m_next) { @@ -404,6 +394,8 @@ size_t ArenaAllocator::getTotalBytesAllocated() // that are unused across all area pages. size_t ArenaAllocator::getTotalBytesUsed() { + assert(isInitialized()); + if (m_lastPage != nullptr) { m_lastPage->m_usedBytes = m_nextFreeByte - m_lastPage->m_contents; @@ -432,42 +424,49 @@ void ArenaAllocator::startup() //------------------------------------------------------------------------ // ArenaAllocator::shutdown: // Performs any necessary teardown for the arena allocator subsystem. -// -// Notes: -// We chose not to call s_pooledAllocator->destroy() and let the memory leak. -// Below is the reason (VSW 600919). -// -// The following race-condition exists during ExitProcess. -// Thread A calls ExitProcess, which causes thread B to terminate. -// Thread B terminated in the middle of reset() -// (through the call-chain of returnPooledAllocator() ==> reset()) -// And then thread A comes along to call s_pooledAllocator->destroy() which will cause the double-free -// of page specified by "temp". -// -// These are possible fixes: -// 1. Thread A tries to get hold on s_isPooledAllocatorInUse lock before -// calling s_theAllocator.destroy(). However, this could cause the deadlock because thread B -// has already gone and therefore it can't release s_isPooledAllocatorInUse. -// 2. Fix the logic in reset() and destroy() to update m_firstPage and m_lastPage in a thread safe way. -// But it needs careful work to make it high performant (e.g. not holding a lock?) -// 3. The scenario of dynamically unloading clrjit.dll cleanly is unimportant at this time. -// We will leak the memory associated with other instances of ArenaAllocator anyway. -// -// Therefore we decided not to call the cleanup code when unloading the jit. void ArenaAllocator::shutdown() +{ + PooledAllocator::shutdown(); +} + +PooledAllocator PooledAllocator::s_pooledAllocator; +LONG PooledAllocator::s_pooledAllocatorState = POOLED_ALLOCATOR_NOTINITIALIZED; + +//------------------------------------------------------------------------ +// PooledAllocator::PooledAllocator: +// Constructs a `PooledAllocator`. +PooledAllocator::PooledAllocator(IEEMemoryManager* memoryManager) + : ArenaAllocator(memoryManager) { } -// The static instance which we try to reuse for all non-simultaneous requests. +//------------------------------------------------------------------------ +// PooledAllocator::shutdown: +// Performs any necessary teardown for the pooled allocator. // -// We try to use this allocator instance as much as possible. It will always -// keep a page handy so small methods won't have to call VirtualAlloc() -// But we may not be able to use it if another thread/reentrant call -// is already using it. -static ArenaAllocator s_theAllocator; +// Notes: +// If the allocator has been initialized and is in use when this method is called, +// it is up to whatever is using the pooled allocator to call `destroy` in order +// to free its memory. +void PooledAllocator::shutdown() +{ + LONG oldState = InterlockedExchange(&s_pooledAllocatorState, POOLED_ALLOCATOR_SHUTDOWN); + switch (oldState) + { + case POOLED_ALLOCATOR_NOTINITIALIZED: + case POOLED_ALLOCATOR_SHUTDOWN: + case POOLED_ALLOCATOR_IN_USE: + return; + + case POOLED_ALLOCATOR_AVAILABLE: + // The pooled allocator was initialized and not in use; we must destroy it. + s_pooledAllocator.destroy(); + break; + } +} //------------------------------------------------------------------------ -// ArenaAllocator::getPooledAllocator: +// PooledAllocator::getPooledAllocator: // Returns the pooled allocator if it is not already in use. // // Arguments: @@ -478,64 +477,103 @@ static ArenaAllocator s_theAllocator; // if it is already in use. // // Notes: -// The returned `ArenaAllocator` should be given back to the pool by -// calling `ArenaAllocator::returnPooledAllocator` when the caller has -// finished using it. -ArenaAllocator* ArenaAllocator::getPooledAllocator(IEEMemoryManager* memoryManager) +// Calling `destroy` on the returned allocator will return it to the +// pool. +ArenaAllocator* PooledAllocator::getPooledAllocator(IEEMemoryManager* memoryManager) { - if (InterlockedExchange(&s_isPooledAllocatorInUse, 1)) + LONG oldState = InterlockedExchange(&s_pooledAllocatorState, POOLED_ALLOCATOR_IN_USE); + switch (oldState) { - // Its being used by another Compiler instance - return nullptr; + case POOLED_ALLOCATOR_IN_USE: + case POOLED_ALLOCATOR_SHUTDOWN: + // Either the allocator is in use or this call raced with a call to `shutdown`. + // Return `nullptr`. + return nullptr; + + case POOLED_ALLOCATOR_AVAILABLE: + if (s_pooledAllocator.m_memoryManager != memoryManager) + { + // The allocator is available, but it was initialized with a different + // memory manager. Release it and return `nullptr`. + InterlockedExchange(&s_pooledAllocatorState, POOLED_ALLOCATOR_AVAILABLE); + return nullptr; + } + + return &s_pooledAllocator; + + case POOLED_ALLOCATOR_NOTINITIALIZED: + { + PooledAllocator allocator(memoryManager); + if (allocator.allocateNewPage(0, false) == nullptr) + { + // Failed to grab the initial memory page. + InterlockedExchange(&s_pooledAllocatorState, POOLED_ALLOCATOR_NOTINITIALIZED); + return nullptr; + } + + s_pooledAllocator = std::move(allocator); + } + + return &s_pooledAllocator; + + default: + assert(!"Unknown pooled allocator state"); + unreached(); } +} - if (s_pooledAllocator == nullptr) +//------------------------------------------------------------------------ +// PooledAllocator::destroy: +// Performs any necessary teardown for an `PooledAllocator` and returns the allocator +// to the pool. +void PooledAllocator::destroy() +{ + assert(isInitialized()); + assert(this == &s_pooledAllocator); + assert(s_pooledAllocatorState == POOLED_ALLOCATOR_IN_USE || s_pooledAllocatorState == POOLED_ALLOCATOR_SHUTDOWN); + assert(m_firstPage != nullptr); + + // Free all but the first allocated page + for (PageDescriptor* page = m_firstPage->m_next, *next; page != nullptr; page = next) { - // Not initialized yet + next = page->m_next; + freeHostMemory(page); + } - bool res = s_theAllocator.initialize(memoryManager, true); - if (!res) - { - // failed to initialize - InterlockedExchange(&s_isPooledAllocatorInUse, 0); - return nullptr; - } + // Reset the relevant state to point back to the first byte of the first page + m_firstPage->m_next = nullptr; + m_lastPage = m_firstPage; + m_nextFreeByte = m_firstPage->m_contents; + m_lastFreeByte = (BYTE*)m_firstPage + m_firstPage->m_pageBytes; + + assert(getTotalBytesAllocated() == s_defaultPageSize); - s_pooledAllocator = &s_theAllocator; - - assert(s_pooledAllocator->getTotalBytesAllocated() == s_defaultPageSize); - s_pooledAllocator->mark(s_pooledAllocatorMark); + // If we've already been shut down, free the first page. Otherwise, return the allocator to the pool. + if (s_pooledAllocatorState == POOLED_ALLOCATOR_SHUTDOWN) + { + ArenaAllocator::destroy(); } else { - if (s_pooledAllocator->m_memoryManager != memoryManager) - { - // already initialize with a different memory manager - InterlockedExchange(&s_isPooledAllocatorInUse, 0); - return nullptr; - } + InterlockedExchange(&s_pooledAllocatorState, POOLED_ALLOCATOR_AVAILABLE); } - - assert(s_pooledAllocator->getTotalBytesAllocated() == s_defaultPageSize); - return s_pooledAllocator; } //------------------------------------------------------------------------ -// ArenaAllocator::returnPooledAllocator: -// Returns the pooled allocator after the callee has finished using it. +// ArenaAllocator::getPooledAllocator: +// Returns the pooled allocator if it is not already in use. // // Arguments: -// allocator - The pooled allocator instance. This must be an instance -// that was obtained by a previous call to -// `ArenaAllocator::getPooledAllocator`. -void ArenaAllocator::returnPooledAllocator(ArenaAllocator* allocator) +// memoryManager: The `IEEMemoryManager` instance in use by the caller. +// +// Return Value: +// A pointer to the pooled allocator if it is available or `nullptr` +// if it is already in use. +// +// Notes: +// Calling `destroy` on the returned allocator will return it to the +// pool. +ArenaAllocator* ArenaAllocator::getPooledAllocator(IEEMemoryManager* memoryManager) { - assert(s_pooledAllocator != nullptr); - assert(s_isPooledAllocatorInUse == 1); - assert(allocator == s_pooledAllocator); - - s_pooledAllocator->reset(s_pooledAllocatorMark); - assert(s_pooledAllocator->getTotalBytesAllocated() == s_defaultPageSize); - - InterlockedExchange(&s_isPooledAllocatorInUse, 0); + return PooledAllocator::getPooledAllocator(memoryManager); } diff --git a/src/jit/alloc.h b/src/jit/alloc.h index aa594336409e..e5aa29251c24 100644 --- a/src/jit/alloc.h +++ b/src/jit/alloc.h @@ -9,16 +9,13 @@ #include "host.h" #endif // defined(_HOST_H_) -struct ArenaAllocator +class ArenaAllocator { private: - struct MarkDescriptor - { - void* m_page; - BYTE* m_next; - BYTE* m_last; - }; + ArenaAllocator(const ArenaAllocator& other) = delete; + ArenaAllocator& operator=(const ArenaAllocator& other) = delete; +protected: struct PageDescriptor { PageDescriptor* m_next; @@ -40,9 +37,8 @@ struct ArenaAllocator }; static size_t s_defaultPageSize; - static ArenaAllocator* s_pooledAllocator; - static MarkDescriptor s_pooledAllocatorMark; - static LONG s_isPooledAllocatorInUse; + + IEEMemoryManager* m_memoryManager; PageDescriptor* m_firstPage; PageDescriptor* m_lastPage; @@ -51,20 +47,25 @@ struct ArenaAllocator BYTE* m_nextFreeByte; BYTE* m_lastFreeByte; - IEEMemoryManager* m_memoryManager; + bool isInitialized(); - void* allocateNewPage(size_t size); - - // The following methods are used for mark/release operation. - void mark(MarkDescriptor& mark); - void reset(MarkDescriptor& mark); + void* allocateNewPage(size_t size, bool canThrow); void* allocateHostMemory(size_t size); void freeHostMemory(void* block); public: - bool initialize(IEEMemoryManager* memoryManager, bool shouldPreallocate); - void destroy(); + ArenaAllocator(); + ArenaAllocator(IEEMemoryManager* memoryManager); + ArenaAllocator& operator=(ArenaAllocator&& other); + + // NOTE: it would be nice to have a destructor on this type to ensure that any value that + // goes out of scope is either uninitialized or has been torn down via a call to + // destroy(), but this interacts badly in methods that use SEH. #3058 tracks + // revisiting EH in the JIT; such a destructor could be added if SEH is removed + // as part of that work. + + virtual void destroy(); #if defined(DEBUG) void* allocateMemory(size_t sz); @@ -76,7 +77,7 @@ struct ArenaAllocator if (m_nextFreeByte > m_lastFreeByte) { - block = allocateNewPage(size); + block = allocateNewPage(size, true); } return block; @@ -92,12 +93,7 @@ struct ArenaAllocator static void startup(); static void shutdown(); - // Gets the pooled allocator if it is available. Returns `nullptr` if the - // pooled allocator is already in use. static ArenaAllocator* getPooledAllocator(IEEMemoryManager* memoryManager); - - // Returns the pooled allocator for use by others. - static void returnPooledAllocator(ArenaAllocator* allocator); }; #endif // _ALLOC_H_ diff --git a/src/jit/compiler.cpp b/src/jit/compiler.cpp index e7ac08ffd66c..faa9bde6bbab 100644 --- a/src/jit/compiler.cpp +++ b/src/jit/compiler.cpp @@ -5622,16 +5622,12 @@ int jitNativeCode ( CORINFO_METHOD_HANDLE methodHnd, { IEEMemoryManager* pMemoryManager = compHnd->getMemoryManager(); - // Try to reuse the pre-inited allocator ? + // Try to reuse the pre-inited allocator pAlloc = ArenaAllocator::getPooledAllocator(pMemoryManager); - if (!pAlloc) + if (pAlloc == nullptr) { - bool res = alloc.initialize(pMemoryManager, false); - if (!res) - { - return CORJIT_OUTOFMEM; - } + alloc = ArenaAllocator(pMemoryManager); pAlloc = &alloc; } } @@ -5737,16 +5733,9 @@ int jitNativeCode ( CORINFO_METHOD_HANDLE methodHnd, } if (pParamOuter->inlineInfo == NULL) - { - // Now free up whichever allocator we were using - if (pParamOuter->pAlloc != pParamOuter->alloc) - { - ArenaAllocator::returnPooledAllocator(pParamOuter->pAlloc); - } - else - { - pParamOuter->alloc->destroy(); - } + { + // Free up the allocator we were using + pParamOuter->pAlloc->destroy(); } } endErrorTrap()