xmrig/src/Mem_unix.cpp

120 lines
3.4 KiB
C++
Raw Normal View History

2017-06-09 21:43:23 +00:00
/* XMRig
* Copyright 2010 Jeff Garzik <jgarzik@pobox.com>
* Copyright 2012-2014 pooler <pooler@litecoinpool.org>
* Copyright 2014 Lucas Jones <https://github.com/lucasjones>
* Copyright 2014-2016 Wolf9466 <https://github.com/OhGodAPet>
* Copyright 2016 Jay D Dee <jayddee246@gmail.com>
2018-03-06 14:34:20 +00:00
* Copyright 2017-2018 XMR-Stak <https://github.com/fireice-uk>, <https://github.com/psychocrypt>
* Copyright 2018 Lee Clagett <https://github.com/vtnerd>
2019-01-14 19:15:36 +00:00
* Copyright 2018-2019 SChernykh <https://github.com/SChernykh>
* Copyright 2016-2019 XMRig <https://github.com/xmrig>, <support@xmrig.com>
2017-06-09 21:43:23 +00:00
*
* 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 <http://www.gnu.org/licenses/>.
*/
2017-06-14 13:11:01 +00:00
#include <stdlib.h>
#include <sys/mman.h>
2019-03-26 18:29:37 +00:00
#include "base/io/log/Log.h"
2018-04-15 04:08:47 +00:00
#include "common/utils/mm_malloc.h"
2018-04-13 02:27:37 +00:00
#include "common/xmrig.h"
2017-06-09 21:43:23 +00:00
#include "crypto/CryptoNight.h"
#include "Mem.h"
2017-06-14 13:11:01 +00:00
2019-03-27 13:03:15 +00:00
#if defined(__APPLE__)
# include <mach/vm_statistics.h>
#endif
2018-04-15 04:08:47 +00:00
void Mem::init(bool enabled)
2017-06-14 13:11:01 +00:00
{
2018-04-15 04:08:47 +00:00
m_enabled = enabled;
}
2017-06-14 13:11:01 +00:00
2018-04-15 04:08:47 +00:00
void Mem::allocate(MemInfo &info, bool enabled)
{
info.hugePages = 0;
2017-08-16 11:21:12 +00:00
if (!enabled) {
2018-04-15 05:58:17 +00:00
info.memory = static_cast<uint8_t*>(_mm_malloc(info.size, 4096));
2017-08-16 11:21:12 +00:00
2018-04-15 04:08:47 +00:00
return;
}
2017-06-14 13:11:01 +00:00
2017-06-14 17:37:59 +00:00
# if defined(__APPLE__)
2018-04-15 04:08:47 +00:00
info.memory = static_cast<uint8_t*>(mmap(0, info.size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANON, VM_FLAGS_SUPERPAGE_SIZE_2MB, 0));
2017-10-20 08:41:08 +00:00
# elif defined(__FreeBSD__)
2018-04-15 04:08:47 +00:00
info.memory = static_cast<uint8_t*>(mmap(0, info.size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS | MAP_ALIGNED_SUPER | MAP_PREFAULT_READ, -1, 0));
2017-06-14 17:37:59 +00:00
# else
2018-04-15 04:08:47 +00:00
info.memory = static_cast<uint8_t*>(mmap(0, info.size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS | MAP_HUGETLB | MAP_POPULATE, 0, 0));
2017-06-14 17:37:59 +00:00
# endif
2018-04-15 04:08:47 +00:00
if (info.memory == MAP_FAILED) {
return allocate(info, false);;
2017-06-14 13:11:01 +00:00
}
2018-04-15 04:08:47 +00:00
info.hugePages = info.pages;
2017-06-14 13:11:01 +00:00
2018-04-15 04:08:47 +00:00
if (madvise(info.memory, info.size, MADV_RANDOM | MADV_WILLNEED) != 0) {
2017-06-14 13:11:01 +00:00
LOG_ERR("madvise failed");
}
2018-04-15 04:08:47 +00:00
if (mlock(info.memory, info.size) == 0) {
2017-06-14 13:11:01 +00:00
m_flags |= Lock;
}
}
2018-04-15 04:08:47 +00:00
void Mem::release(MemInfo &info)
2017-06-14 13:11:01 +00:00
{
2018-04-15 04:08:47 +00:00
if (info.hugePages) {
2017-06-14 13:11:01 +00:00
if (m_flags & Lock) {
2018-04-15 04:08:47 +00:00
munlock(info.memory, info.size);
2017-06-14 13:11:01 +00:00
}
2018-04-15 04:08:47 +00:00
munmap(info.memory, info.size);
2017-06-14 13:11:01 +00:00
}
else {
2018-04-15 04:08:47 +00:00
_mm_free(info.memory);
2017-06-14 13:11:01 +00:00
}
}
2019-01-14 19:15:36 +00:00
void *Mem::allocateExecutableMemory(size_t size)
{
# if defined(__APPLE__)
return mmap(0, size, PROT_READ | PROT_WRITE | PROT_EXEC, MAP_PRIVATE | MAP_ANON, -1, 0);
# else
return mmap(0, size, PROT_READ | PROT_WRITE | PROT_EXEC, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
# endif
}
void Mem::protectExecutableMemory(void *p, size_t size)
{
mprotect(p, size, PROT_READ | PROT_EXEC);
}
2019-01-14 19:15:36 +00:00
void Mem::flushInstructionCache(void *p, size_t size)
{
# ifndef __FreeBSD__
__builtin___clear_cache(reinterpret_cast<char*>(p), reinterpret_cast<char*>(p) + size);
# endif
}