xmrig/src/common/Platform_win.cpp

205 lines
5 KiB
C++
Raw Normal View History

2017-06-04 17:52:21 +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-11-20 00:24:14 +00:00
* Copyright 2017-2018 XMR-Stak <https://github.com/fireice-uk>, <https://github.com/psychocrypt>
* Copyright 2018 SChernykh <https://github.com/SChernykh>
2019-01-09 09:43:36 +00:00
* Copyright 2016-2019 XMRig <https://github.com/xmrig>, <support@xmrig.com>
2017-06-04 17:52:21 +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/>.
*/
2018-11-20 00:24:14 +00:00
#include <algorithm>
2017-06-13 03:31:25 +00:00
#include <winsock2.h>
2017-06-04 17:52:21 +00:00
#include <windows.h>
2017-08-15 00:04:46 +00:00
#include <uv.h>
2017-06-04 17:52:21 +00:00
2018-04-13 02:27:37 +00:00
#include "log/Log.h"
2017-08-15 00:04:46 +00:00
#include "Platform.h"
2017-06-04 17:52:21 +00:00
#include "version.h"
2018-04-13 02:27:37 +00:00
2017-08-17 04:33:21 +00:00
#ifdef XMRIG_NVIDIA_PROJECT
# include "nvidia/cryptonight.h"
#endif
2017-06-04 17:52:21 +00:00
2019-01-09 09:43:36 +00:00
#ifdef XMRIG_AMD_PROJECT
2018-11-20 00:24:14 +00:00
static uint32_t timerResolution = 0;
2019-01-09 09:43:36 +00:00
#endif
2018-11-20 00:24:14 +00:00
2017-06-04 17:52:21 +00:00
static inline OSVERSIONINFOEX winOsVersion()
{
typedef NTSTATUS (NTAPI *RtlGetVersionFunction)(LPOSVERSIONINFO);
OSVERSIONINFOEX result = { sizeof(OSVERSIONINFOEX), 0, 0, 0, 0, {'\0'}, 0, 0, 0, 0, 0};
HMODULE ntdll = GetModuleHandleW(L"ntdll.dll");
if (ntdll ) {
RtlGetVersionFunction pRtlGetVersion = reinterpret_cast<RtlGetVersionFunction>(GetProcAddress(ntdll, "RtlGetVersion"));
if (pRtlGetVersion) {
pRtlGetVersion((LPOSVERSIONINFO) &result);
}
}
return result;
}
2018-09-16 00:06:54 +00:00
char *Platform::createUserAgent()
2017-06-04 17:52:21 +00:00
{
const auto osver = winOsVersion();
constexpr const size_t max = 256;
2017-06-04 17:52:21 +00:00
char *buf = new char[max]();
2017-06-13 03:31:25 +00:00
int length = snprintf(buf, max, "%s/%s (Windows NT %lu.%lu", APP_NAME, APP_VERSION, osver.dwMajorVersion, osver.dwMinorVersion);
2017-06-04 17:52:21 +00:00
2017-06-13 03:31:25 +00:00
# if defined(__x86_64__) || defined(_M_AMD64)
length += snprintf(buf + length, max - length, "; Win64; x64) libuv/%s", uv_version_string());
2017-06-04 17:52:21 +00:00
# else
2017-06-13 03:31:25 +00:00
length += snprintf(buf + length, max - length, ") libuv/%s", uv_version_string());
# endif
2017-08-17 04:33:21 +00:00
# ifdef XMRIG_NVIDIA_PROJECT
const int cudaVersion = cuda_get_runtime_version();
length += snprintf(buf + length, max - length, " CUDA/%d.%d", cudaVersion / 1000, cudaVersion % 100);
# endif
2017-06-13 03:31:25 +00:00
# ifdef __GNUC__
length += snprintf(buf + length, max - length, " gcc/%d.%d.%d", __GNUC__, __GNUC_MINOR__, __GNUC_PATCHLEVEL__);
2017-06-14 13:11:01 +00:00
# elif _MSC_VER
2017-06-13 03:31:25 +00:00
length += snprintf(buf + length, max - length, " msvc/%d", MSVC_VERSION);
2017-06-04 17:52:21 +00:00
# endif
return buf;
}
2017-08-15 00:04:46 +00:00
2018-04-13 02:27:37 +00:00
bool Platform::setThreadAffinity(uint64_t cpu_id)
2017-08-15 00:04:46 +00:00
{
2018-04-13 02:27:37 +00:00
if (cpu_id >= 64) {
LOG_ERR("Unable to set affinity. Windows supports only affinity up to 63.");
}
return SetThreadAffinityMask(GetCurrentThread(), 1ULL << cpu_id) != 0;
2017-08-15 00:04:46 +00:00
}
2018-11-20 00:24:14 +00:00
uint32_t Platform::setTimerResolution(uint32_t resolution)
{
# ifdef XMRIG_AMD_PROJECT
TIMECAPS tc;
if (timeGetDevCaps(&tc, sizeof(TIMECAPS)) != TIMERR_NOERROR) {
return 0;
}
timerResolution = std::min<uint32_t>(std::max<uint32_t>(tc.wPeriodMin, resolution), tc.wPeriodMax);
return timeBeginPeriod(timerResolution) == TIMERR_NOERROR ? timerResolution : 0;
# else
return resolution;
# endif
}
void Platform::restoreTimerResolution()
{
# ifdef XMRIG_AMD_PROJECT
if (timerResolution) {
timeEndPeriod(timerResolution);
}
# endif
}
2017-08-15 05:19:55 +00:00
void Platform::setProcessPriority(int priority)
{
if (priority == -1) {
return;
}
DWORD prio = IDLE_PRIORITY_CLASS;
switch (priority)
{
case 1:
prio = BELOW_NORMAL_PRIORITY_CLASS;
break;
case 2:
prio = NORMAL_PRIORITY_CLASS;
break;
case 3:
prio = ABOVE_NORMAL_PRIORITY_CLASS;
break;
case 4:
prio = HIGH_PRIORITY_CLASS;
break;
case 5:
prio = REALTIME_PRIORITY_CLASS;
2018-11-20 00:24:14 +00:00
break;
2017-08-15 05:19:55 +00:00
default:
break;
}
SetPriorityClass(GetCurrentProcess(), prio);
}
void Platform::setThreadPriority(int priority)
{
if (priority == -1) {
return;
}
int prio = THREAD_PRIORITY_IDLE;
switch (priority)
{
case 1:
prio = THREAD_PRIORITY_BELOW_NORMAL;
break;
case 2:
prio = THREAD_PRIORITY_NORMAL;
break;
case 3:
prio = THREAD_PRIORITY_ABOVE_NORMAL;
break;
case 4:
prio = THREAD_PRIORITY_HIGHEST;
break;
case 5:
prio = THREAD_PRIORITY_TIME_CRITICAL;
break;
default:
break;
}
SetThreadPriority(GetCurrentThread(), prio);
}