2 * Copyright (C) 2009 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "ExecutableAllocator.h"
29 #include "JSCInlines.h"
31 #if ENABLE(EXECUTABLE_ALLOCATOR_FIXED)
33 #include "CodeProfiling.h"
34 #include "ExecutableAllocationFuzz.h"
39 #include <wtf/MetaAllocator.h>
40 #include <wtf/PageReservation.h>
41 #include <wtf/VMTags.h>
55 uintptr_t startOfFixedExecutableMemoryPool;
57 class FixedVMPoolExecutableAllocator : public MetaAllocator {
58 WTF_MAKE_FAST_ALLOCATED;
60 FixedVMPoolExecutableAllocator()
61 : MetaAllocator(jitAllocationGranule) // round up all allocations to 32 bytes
63 m_reservation = PageReservation::reserveWithGuardPages(fixedExecutableMemoryPoolSize, OSAllocator::JSJITCodePages, EXECUTABLE_POOL_WRITABLE, true);
65 ASSERT(m_reservation.size() == fixedExecutableMemoryPoolSize);
66 addFreshFreeSpace(m_reservation.base(), m_reservation.size());
68 startOfFixedExecutableMemoryPool = reinterpret_cast<uintptr_t>(m_reservation.base());
72 virtual ~FixedVMPoolExecutableAllocator();
75 virtual void* allocateNewSpace(size_t&) override
77 // We're operating in a fixed pool, so new allocation is always prohibited.
81 virtual void notifyNeedPage(void* page) override
83 #if USE(MADV_FREE_FOR_JIT_MEMORY)
86 m_reservation.commit(page, pageSize());
90 virtual void notifyPageIsFree(void* page) override
92 #if USE(MADV_FREE_FOR_JIT_MEMORY)
94 int result = madvise(page, pageSize(), MADV_FREE);
98 if (errno != EAGAIN) {
99 RELEASE_ASSERT_NOT_REACHED(); // In debug mode, this should be a hard failure.
100 break; // In release mode, we should just ignore the error - not returning memory to the OS is better than crashing, especially since we _will_ be able to reuse the memory internally anyway.
104 m_reservation.decommit(page, pageSize());
109 PageReservation m_reservation;
112 static FixedVMPoolExecutableAllocator* allocator;
114 void ExecutableAllocator::initializeAllocator()
117 allocator = new FixedVMPoolExecutableAllocator();
118 CodeProfiling::notifyAllocator(allocator);
121 ExecutableAllocator::ExecutableAllocator(VM&)
126 ExecutableAllocator::~ExecutableAllocator()
130 FixedVMPoolExecutableAllocator::~FixedVMPoolExecutableAllocator()
132 m_reservation.deallocate();
135 bool ExecutableAllocator::isValid() const
137 return !!allocator->bytesReserved();
140 bool ExecutableAllocator::underMemoryPressure()
142 MetaAllocator::Statistics statistics = allocator->currentStatistics();
143 return statistics.bytesAllocated > statistics.bytesReserved / 2;
146 double ExecutableAllocator::memoryPressureMultiplier(size_t addedMemoryUsage)
148 MetaAllocator::Statistics statistics = allocator->currentStatistics();
149 ASSERT(statistics.bytesAllocated <= statistics.bytesReserved);
150 size_t bytesAllocated = statistics.bytesAllocated + addedMemoryUsage;
151 if (bytesAllocated >= statistics.bytesReserved)
152 bytesAllocated = statistics.bytesReserved;
154 size_t divisor = statistics.bytesReserved - bytesAllocated;
156 result = static_cast<double>(statistics.bytesReserved) / divisor;
162 PassRefPtr<ExecutableMemoryHandle> ExecutableAllocator::allocate(VM&, size_t sizeInBytes, void* ownerUID, JITCompilationEffort effort)
164 if (effort != JITCompilationCanFail && Options::reportMustSucceedExecutableAllocations()) {
165 dataLog("Allocating ", sizeInBytes, " bytes of executable memory with JITCompilationMustSucceed.\n");
166 WTFReportBacktrace();
169 if (effort == JITCompilationCanFail
170 && doExecutableAllocationFuzzingIfEnabled() == PretendToFailExecutableAllocation)
173 RefPtr<ExecutableMemoryHandle> result = allocator->allocate(sizeInBytes, ownerUID);
175 if (effort != JITCompilationCanFail) {
176 dataLog("Ran out of executable memory while allocating ", sizeInBytes, " bytes.\n");
181 return result.release();
184 size_t ExecutableAllocator::committedByteCount()
186 return allocator->bytesCommitted();
189 #if ENABLE(META_ALLOCATOR_PROFILE)
190 void ExecutableAllocator::dumpProfile()
192 allocator->dumpProfile();
199 #endif // ENABLE(EXECUTABLE_ALLOCATOR_FIXED)