// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #include = size)) throw std::bad_alloc(); return ret; } uint ircd::allocator::state::allocate(std::nothrow_t, const size_type &n, const uint &hint) { const auto next(this->next(n)); if(unlikely(next >= size)) // No block of n was found anywhere (next is past-the-end) return next; for(size_t i(0); i < n; ++i) { assert(!test(next + i)); bts(next + i); } last = next + n; return next; } uint ircd::allocator::state::next(const size_t &n) const { uint ret(last), rem(n); for(; ret < size && rem; ++ret) if(test(ret)) rem = n; else --rem; if(likely(!rem)) return ret - n; for(ret = 0, rem = n; ret < last && rem; ++ret) if(test(ret)) rem = n; else --rem; if(unlikely(rem)) // The allocator should throw std::bad_alloc if !rem return size; return ret - n; } bool ircd::allocator::state::available(const size_t &n) const { return this->next(n) < size; } // // allocator::scope // decltype(ircd::allocator::scope::current) ircd::allocator::scope::current; ircd::allocator::scope::scope(alloc_closure ac, realloc_closure rc, free_closure fc) :theirs { current } ,user_alloc { std::move(ac) } ,user_realloc { std::move(rc) } ,user_free { std::move(fc) } { // If an allocator::scope instance already exists somewhere // up the stack, *current will already be set. We only install // our global hook handlers at the first instance ctor and // uninstall it after that first instance dtors. if(!current) hook_init(); current = this; } ircd::allocator::scope::~scope() noexcept { assert(current); current = theirs; // Reinstall the pre-existing hooks after our last scope instance // has destructed (the first to have constructed). We know this when // current becomes null. if(!current) hook_fini(); } void __attribute__((weak)) ircd::allocator::scope::hook_init() noexcept { } void __attribute__((weak)) ircd::allocator::scope::hook_fini() noexcept { } // // allocator::profile // thread_local ircd::allocator::profile ircd::allocator::profile::this_thread {}; ircd::allocator::profile ircd::allocator::operator-(const profile &a, const profile &b) { profile ret(a); ret -= b; return ret; } ircd::allocator::profile ircd::allocator::operator+(const profile &a, const profile &b) { profile ret(a); ret += b; return ret; } ircd::allocator::profile & ircd::allocator::operator-=(profile &a, const profile &b) { a.alloc_count -= b.alloc_count; a.free_count -= b.free_count; a.alloc_bytes -= b.alloc_bytes; a.free_bytes -= b.free_bytes; return a; } ircd::allocator::profile & ircd::allocator::operator+=(profile &a, const profile &b) { a.alloc_count += b.alloc_count; a.free_count += b.free_count; a.alloc_bytes += b.alloc_bytes; a.free_bytes += b.free_bytes; return a; } // // aligned_alloc // std::unique_ptr ircd::allocator::aligned_alloc(const size_t &alignment_, const size_t &size_) { static const size_t &align_default { 16 }; const size_t &alignment { alignment_?: align_default }; const size_t &size { size_ % alignment == 0? size_: size_ + (alignment - (size_ % alignment)) }; assert(size % alignment == 0); assert(size < size_ + alignment); assert(alignment % sizeof(void *) == 0); void *ret; switch(int errc(::posix_memalign(&ret, alignment, size)); errc) { case 0: break; case int(std::errc::not_enough_memory): throw std::bad_alloc{}; default: throw std::system_error { errc, std::system_category() }; } assert(ret != nullptr); assert(uintptr_t(ret) % alignment == 0); #ifdef RB_PROF_ALLOC auto &this_thread(ircd::allocator::profile::this_thread); this_thread.alloc_bytes += size; this_thread.alloc_count++; #endif return { reinterpret_cast(ret), &std::free }; } // // resource limits // #if defined(HAVE_SYS_RESOURCE_H) && defined(RLIMIT_MEMLOCK) size_t ircd::allocator::rlimit_memlock(const size_t &req) try { rlimit rlim {0}; rlim.rlim_cur = req; syscall(setrlimit, RLIMIT_MEMLOCK, &rlim); char pbuf[48]; log::info { "Raised resource limit for locked memory to %s", req != -1UL? pretty(pbuf, iec(req)): "unlimited"_sv, }; return rlim.rlim_cur; } catch(const std::system_error &e) { char pbuf[48]; log::warning { "Failed to raise resource limit for locked memory to %s :%s", req != -1UL? pretty(pbuf, iec(req)): "unlimited"_sv, e.what(), }; return rlimit_memlock(); } #else size_t ircd::allocator::rlimit_memlock(const size_t &req) { return 0; } #endif #if defined(HAVE_SYS_RESOURCE_H) && defined(RLIMIT_MEMLOCK) size_t ircd::allocator::rlimit_memlock() { rlimit rlim; syscall(getrlimit, RLIMIT_MEMLOCK, &rlim); return rlim.rlim_cur; } #else size_t ircd::allocator::rlimit_memlock() { return 0; } #endif #if defined(HAVE_SYS_RESOURCE_H) && defined(RLIMIT_DATA) size_t ircd::allocator::rlimit_data() { rlimit rlim; syscall(getrlimit, RLIMIT_DATA, &rlim); return rlim.rlim_cur; } #else size_t ircd::allocator::rlimit_data() { return 0; } #endif #if defined(HAVE_SYS_RESOURCE_H) && defined(RLIMIT_AS) size_t ircd::allocator::rlimit_as() { rlimit rlim; syscall(getrlimit, RLIMIT_AS, &rlim); return rlim.rlim_cur; } #else size_t ircd::allocator::rlimit_as() { return 0; } #endif // // Developer profiling // #ifdef RB_PROF_ALLOC // -------------------------------------------------- void * __attribute__((alloc_size(1), malloc, returns_nonnull)) operator new(const size_t size) { void *const &ptr(::malloc(size)); if(unlikely(!ptr)) throw std::bad_alloc(); auto &this_thread(ircd::allocator::profile::this_thread); this_thread.alloc_bytes += size; this_thread.alloc_count++; return ptr; } void operator delete(void *const ptr) noexcept { ::free(ptr); auto &this_thread(ircd::allocator::profile::this_thread); this_thread.free_count++; } void operator delete(void *const ptr, const size_t size) noexcept { ::free(ptr); auto &this_thread(ircd::allocator::profile::this_thread); this_thread.free_bytes += size; this_thread.free_count++; } #endif // RB_PROF_ALLOC -------------------------------------------------- // // Linker symbol wrapping hook // extern "C" [[gnu::weak]] void *__real_malloc(size_t size); extern "C" [[gnu::weak]] void *__real_calloc(size_t nmemb, size_t size); extern "C" [[gnu::weak]] void *__real_realloc(void *ptr, size_t size); extern "C" [[gnu::weak]] void __real_free(void *ptr); extern "C" void * __wrap_malloc(size_t size) { void *const &ptr(::__real_malloc(size)); if(unlikely(!ptr)) throw std::bad_alloc(); auto &this_thread(ircd::allocator::profile::this_thread); this_thread.alloc_bytes += size; this_thread.alloc_count++; return ptr; } extern "C" void * __wrap_calloc(size_t nmemb, size_t size) { void *const &ptr(::__real_calloc(nmemb, size)); if(unlikely(!ptr)) throw std::bad_alloc(); auto &this_thread(ircd::allocator::profile::this_thread); this_thread.alloc_bytes += nmemb * size; this_thread.alloc_count++; return __real_calloc(nmemb, size); } extern "C" void * __wrap_realloc(void *ptr, size_t size) { void *const &ret(::__real_realloc(ptr, size)); if(unlikely(!ret)) throw std::bad_alloc(); auto &this_thread(ircd::allocator::profile::this_thread); this_thread.alloc_bytes += size; this_thread.alloc_count++; return ret; } extern "C" void __wrap_free(void *ptr) { __real_free(ptr); auto &this_thread(ircd::allocator::profile::this_thread); this_thread.free_bytes += 0UL; //TODO: XXX this_thread.free_count++; }