blob: 26bfc233a981be60a58f8316cb9d3a8148b5e8af [file] [log] [blame]
/*
* Copyright (C) 2017-2018 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "IsoTLS.h"
#include "Environment.h"
#include "IsoTLSEntryInlines.h"
#include "IsoTLSInlines.h"
#include "IsoTLSLayout.h"
#include <stdio.h>
namespace bmalloc {
IsoTLS::MallocFallbackState IsoTLS::s_mallocFallbackState;
#if !HAVE_PTHREAD_MACHDEP_H
bool IsoTLS::s_didInitialize;
pthread_key_t IsoTLS::s_tlsKey;
#endif
void IsoTLS::scavenge()
{
if (IsoTLS* tls = get()) {
tls->forEachEntry(
[&] (IsoTLSEntry* entry, void* data) {
entry->scavenge(data);
});
}
}
IsoTLS::IsoTLS()
{
BASSERT(!Environment::get()->isDebugHeapEnabled());
}
IsoTLS* IsoTLS::ensureEntries(unsigned offset)
{
RELEASE_BASSERT(!get() || offset >= get()->m_extent);
static std::once_flag onceFlag;
std::call_once(
onceFlag,
[] () {
setvbuf(stderr, NULL, _IONBF, 0);
#if HAVE_PTHREAD_MACHDEP_H
pthread_key_init_np(tlsKey, destructor);
#else
int error = pthread_key_create(&s_tlsKey, destructor);
if (error)
BCRASH();
s_didInitialize = true;
#endif
});
IsoTLS* tls = get();
IsoTLSLayout& layout = *IsoTLSLayout::get();
IsoTLSEntry* oldLastEntry = tls ? tls->m_lastEntry : nullptr;
RELEASE_BASSERT(!oldLastEntry || oldLastEntry->offset() < offset);
IsoTLSEntry* startEntry = oldLastEntry ? oldLastEntry->m_next : layout.head();
RELEASE_BASSERT(startEntry);
IsoTLSEntry* targetEntry = startEntry;
for (;;) {
RELEASE_BASSERT(targetEntry);
RELEASE_BASSERT(targetEntry->offset() <= offset);
if (targetEntry->offset() == offset)
break;
targetEntry = targetEntry->m_next;
}
RELEASE_BASSERT(targetEntry);
size_t requiredCapacity = targetEntry->extent();
if (!tls || requiredCapacity > tls->m_capacity) {
size_t requiredSize = sizeForCapacity(requiredCapacity);
size_t goodSize = roundUpToMultipleOf(vmPageSize(), requiredSize);
size_t goodCapacity = capacityForSize(goodSize);
void* memory = vmAllocate(goodSize, VMTag::IsoHeap);
IsoTLS* newTLS = new (memory) IsoTLS();
newTLS->m_capacity = goodCapacity;
if (tls) {
RELEASE_BASSERT(oldLastEntry);
RELEASE_BASSERT(layout.head());
layout.head()->walkUpToInclusive(
oldLastEntry,
[&] (IsoTLSEntry* entry) {
void* src = tls->m_data + entry->offset();
void* dst = newTLS->m_data + entry->offset();
entry->move(src, dst);
entry->destruct(src);
});
size_t oldSize = tls->size();
tls->~IsoTLS();
vmDeallocate(tls, oldSize);
}
tls = newTLS;
set(tls);
}
startEntry->walkUpToInclusive(
targetEntry,
[&] (IsoTLSEntry* entry) {
entry->construct(tls->m_data + entry->offset());
});
tls->m_lastEntry = targetEntry;
tls->m_extent = targetEntry->extent();
return tls;
}
void IsoTLS::destructor(void* arg)
{
IsoTLS* tls = static_cast<IsoTLS*>(arg);
RELEASE_BASSERT(tls);
tls->forEachEntry(
[&] (IsoTLSEntry* entry, void* data) {
entry->scavenge(data);
entry->destruct(data);
});
size_t oldSize = tls->size();
tls->~IsoTLS();
vmDeallocate(tls, oldSize);
}
size_t IsoTLS::sizeForCapacity(unsigned capacity)
{
return BOFFSETOF(IsoTLS, m_data) + capacity;
}
unsigned IsoTLS::capacityForSize(size_t size)
{
return size - sizeForCapacity(0);
}
size_t IsoTLS::size()
{
return sizeForCapacity(m_capacity);
}
template<typename Func>
void IsoTLS::forEachEntry(const Func& func)
{
if (!m_lastEntry)
return;
IsoTLSLayout::get()->head()->walkUpToInclusive(
m_lastEntry,
[&] (IsoTLSEntry* entry) {
func(entry, m_data + entry->offset());
});
}
void IsoTLS::determineMallocFallbackState()
{
static std::once_flag onceFlag;
std::call_once(
onceFlag,
[] {
if (s_mallocFallbackState != MallocFallbackState::Undecided)
return;
if (Environment::get()->isDebugHeapEnabled()) {
s_mallocFallbackState = MallocFallbackState::FallBackToMalloc;
return;
}
const char* env = getenv("bmalloc_IsoHeap");
if (env && (!strcasecmp(env, "false") || !strcasecmp(env, "no") || !strcmp(env, "0")))
s_mallocFallbackState = MallocFallbackState::FallBackToMalloc;
else
s_mallocFallbackState = MallocFallbackState::DoNotFallBack;
});
}
} // namespace bmalloc