/* XMRig * Copyright 2010 Jeff Garzik * Copyright 2012-2014 pooler * Copyright 2014 Lucas Jones * Copyright 2014-2016 Wolf9466 * Copyright 2016 Jay D Dee * Copyright 2017-2018 XMR-Stak , * Copyright 2018 Lee Clagett * Copyright 2018-2019 SChernykh * Copyright 2018-2019 tevador * Copyright 2016-2019 XMRig , * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "crypto/common/VirtualMemory.h" #include "backend/cpu/Cpu.h" #include "base/io/log/Log.h" #include "crypto/common/MemoryPool.h" #include "crypto/common/portable/mm_malloc.h" #include #include namespace xmrig { static IMemoryPool *pool = nullptr; static std::mutex mutex; } // namespace xmrig xmrig::VirtualMemory::VirtualMemory(size_t size, bool hugePages, bool usePool, uint32_t node, size_t alignSize) : m_size(align(size)), m_node(node) { if (usePool) { std::lock_guard lock(mutex); if (hugePages && !pool->isHugePages(node) && allocateLargePagesMemory()) { return; } m_scratchpad = pool->get(m_size, node); if (m_scratchpad) { m_flags.set(FLAG_HUGEPAGES, pool->isHugePages(node)); m_flags.set(FLAG_EXTERNAL, true); return; } } if (hugePages && allocateLargePagesMemory()) { return; } m_scratchpad = static_cast(_mm_malloc(m_size, alignSize)); } xmrig::VirtualMemory::~VirtualMemory() { if (!m_scratchpad) { return; } if (m_flags.test(FLAG_EXTERNAL)) { std::lock_guard lock(mutex); pool->release(m_node); } if (isHugePages()) { freeLargePagesMemory(); } else { _mm_free(m_scratchpad); } } #ifndef XMRIG_FEATURE_HWLOC uint32_t xmrig::VirtualMemory::bindToNUMANode(int64_t) { return 0; } #endif void xmrig::VirtualMemory::destroy() { delete pool; } void xmrig::VirtualMemory::init(bool hugePages, int poolSize) { if (!pool) { osInit(); } pool = new MemoryPool(poolSize < 0 ? Cpu::info()->threads() : poolSize, hugePages); }