mirror of
https://github.com/anrieff/libcpuid
synced 2025-01-23 20:06:41 +00:00
Restore previous thread CPU affinity before returning from cpuid_get_all_raw_data() (#184)
* Set thread instead of process CPU affinity in pre Windows 7 case * Restore previous thread CPU affinity before returning from cpuid_get_all_raw_data()
This commit is contained in:
parent
c7c2bd3619
commit
54ed9a0a17
1 changed files with 160 additions and 6 deletions
|
@ -40,13 +40,15 @@
|
||||||
/* Implementation: */
|
/* Implementation: */
|
||||||
|
|
||||||
#if defined(__STDC_VERSION__) && __STDC_VERSION__ == 201112L
|
#if defined(__STDC_VERSION__) && __STDC_VERSION__ == 201112L
|
||||||
_Thread_local int _libcpuid_errno = ERR_OK;
|
#define INTERNAL_SCOPE _Thread_local
|
||||||
#elif defined(__GNUC__) // Also works for clang
|
#elif defined(__GNUC__) // Also works for clang
|
||||||
__thread int _libcpuid_errno = ERR_OK;
|
#define INTERNAL_SCOPE __thread
|
||||||
#else
|
#else
|
||||||
static int _libcpuid_errno = ERR_OK;
|
#define INTERNAL_SCOPE static
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
INTERNAL_SCOPE int _libcpuid_errno = ERR_OK;
|
||||||
|
|
||||||
int set_error(cpu_error_t err)
|
int set_error(cpu_error_t err)
|
||||||
{
|
{
|
||||||
_libcpuid_errno = (int) err;
|
_libcpuid_errno = (int) err;
|
||||||
|
@ -173,6 +175,21 @@ static int get_total_cpus(void)
|
||||||
}
|
}
|
||||||
#define GET_TOTAL_CPUS_DEFINED
|
#define GET_TOTAL_CPUS_DEFINED
|
||||||
|
|
||||||
|
INTERNAL_SCOPE thread_affinity_policy_data_t saved_affinity;
|
||||||
|
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
mach_msg_type_number_t count = THREAD_AFFINITY_POLICY_COUNT;
|
||||||
|
boolean_t get_default = false;
|
||||||
|
return thread_policy_get(mach_thread_self(), THREAD_AFFINITY_POLICY, (thread_policy_t) &saved_affinity, &count, &get_default) == KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
return thread_policy_set(mach_thread_self(), THREAD_AFFINITY_POLICY, (thread_policy_t) &saved_affinity, THREAD_AFFINITY_POLICY_COUNT) == KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
#define PRESERVE_CPU_AFFINITY
|
||||||
|
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
thread_affinity_policy_data_t ap;
|
thread_affinity_policy_data_t ap;
|
||||||
|
@ -196,6 +213,52 @@ static int get_total_cpus(void)
|
||||||
}
|
}
|
||||||
#define GET_TOTAL_CPUS_DEFINED
|
#define GET_TOTAL_CPUS_DEFINED
|
||||||
|
|
||||||
|
#if (_WIN32_WINNT >= 0x0601)
|
||||||
|
INTERNAL_SCOPE GROUP_AFFINITY savedGroupAffinity;
|
||||||
|
#else
|
||||||
|
INTERNAL_SCOPE DWORD_PTR savedAffinityMask = 0;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
#if (_WIN32_WINNT >= 0x0601)
|
||||||
|
HANDLE thread = GetCurrentThread();
|
||||||
|
return GetThreadGroupAffinity(thread, &savedGroupAffinity);
|
||||||
|
#else
|
||||||
|
/* Credits to https://stackoverflow.com/questions/6601862/query-thread-not-process-processor-affinity#6601917 */
|
||||||
|
HANDLE thread = GetCurrentThread();
|
||||||
|
DWORD_PTR threadAffinityMask = 1;
|
||||||
|
while (threadAffinityMask) {
|
||||||
|
savedAffinityMask = SetThreadAffinityMask(thread, threadAffinityMask);
|
||||||
|
if(savedAffinityMask)
|
||||||
|
return SetThreadAffinityMask(thread, savedAffinityMask);
|
||||||
|
else if (GetLastError() != ERROR_INVALID_PARAMETER)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
threadAffinityMask <<= 1; // try next CPU
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
#if (_WIN32_WINNT >= 0x0601)
|
||||||
|
if (!savedGroupAffinity.Mask)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
HANDLE thread = GetCurrentThread();
|
||||||
|
return SetThreadGroupAffinity(thread, &savedGroupAffinity, NULL);
|
||||||
|
#else
|
||||||
|
if (!savedAffinityMask)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
HANDLE thread = GetCurrentThread();
|
||||||
|
return SetThreadAffinityMask(thread, savedAffinityMask);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
#define PRESERVE_CPU_AFFINITY
|
||||||
|
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
/* Credits to https://github.com/PolygonTek/BlueshiftEngine/blob/fbc374cbc391e1147c744649f405a66a27c35d89/Source/Runtime/Private/Platform/Windows/PlatformWinThread.cpp#L27 */
|
/* Credits to https://github.com/PolygonTek/BlueshiftEngine/blob/fbc374cbc391e1147c744649f405a66a27c35d89/Source/Runtime/Private/Platform/Windows/PlatformWinThread.cpp#L27 */
|
||||||
|
@ -229,9 +292,9 @@ static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
warnf("set_cpu_affinity for logical CPU %u is not supported in this operating system.\n", logical_cpu);
|
warnf("set_cpu_affinity for logical CPU %u is not supported in this operating system.\n", logical_cpu);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
HANDLE process = GetCurrentProcess();
|
HANDLE thread = GetCurrentThread();
|
||||||
DWORD_PTR processAffinityMask = 1ULL << logical_cpu;
|
DWORD_PTR threadAffinityMask = 1ULL << logical_cpu;
|
||||||
return SetProcessAffinityMask(process, processAffinityMask);
|
return SetThreadAffinityMask(thread, threadAffinityMask);
|
||||||
#endif /* (_WIN32_WINNT >= 0x0601) */
|
#endif /* (_WIN32_WINNT >= 0x0601) */
|
||||||
}
|
}
|
||||||
#define SET_CPU_AFFINITY
|
#define SET_CPU_AFFINITY
|
||||||
|
@ -262,6 +325,19 @@ static int get_total_cpus(void)
|
||||||
#if defined linux || defined __linux__
|
#if defined linux || defined __linux__
|
||||||
#include <sched.h>
|
#include <sched.h>
|
||||||
|
|
||||||
|
INTERNAL_SCOPE cpu_set_t saved_affinity;
|
||||||
|
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
return sched_getaffinity(0, sizeof(saved_affinity), &saved_affinity) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
return sched_setaffinity(0, sizeof(saved_affinity), &saved_affinity) == 0;
|
||||||
|
}
|
||||||
|
#define PRESERVE_CPU_AFFINITY
|
||||||
|
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
cpu_set_t cpuset;
|
cpu_set_t cpuset;
|
||||||
|
@ -277,6 +353,19 @@ static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
#include <sys/processor.h>
|
#include <sys/processor.h>
|
||||||
#include <sys/procset.h>
|
#include <sys/procset.h>
|
||||||
|
|
||||||
|
INTERNAL_SCOPE processorid_t saved_binding = PBIND_NONE;
|
||||||
|
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
return processor_bind(P_LWPID, P_MYID, PBIND_QUERY, &saved_binding) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
return processor_bind(P_LWPID, P_MYID, saved_binding, NULL) == 0;
|
||||||
|
}
|
||||||
|
#define PRESERVE_CPU_AFFINITY
|
||||||
|
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
if (logical_cpu > (sizeof(processorid_t) * 8)) {
|
if (logical_cpu > (sizeof(processorid_t) * 8)) {
|
||||||
|
@ -307,6 +396,19 @@ static int get_total_cpus(void)
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/cpuset.h>
|
#include <sys/cpuset.h>
|
||||||
|
|
||||||
|
INTERNAL_SCOPE cpuset_t saved_affinity;
|
||||||
|
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
return cpuset_getaffinity(CPU_LEVEL_WHICH, CPU_WHICH_TID, -1, sizeof(saved_affinity), &saved_affinity) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
return cpuset_setaffinity(CPU_LEVEL_WHICH, CPU_WHICH_TID, -1, sizeof(saved_affinity), &saved_affinity) == 0;
|
||||||
|
}
|
||||||
|
#define PRESERVE_CPU_AFFINITY
|
||||||
|
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
cpuset_t cpuset;
|
cpuset_t cpuset;
|
||||||
|
@ -321,6 +423,19 @@ static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
#include <pthread.h>
|
#include <pthread.h>
|
||||||
#include <pthread_np.h>
|
#include <pthread_np.h>
|
||||||
|
|
||||||
|
INTERNAL_SCOPE cpuset_t saved_affinity;
|
||||||
|
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
return pthread_getaffinity_np(pthread_self(), sizeof(saved_affinity), &saved_affinity) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
return pthread_setaffinity_np(pthread_self(), sizeof(saved_affinity), &saved_affinity) == 0;
|
||||||
|
}
|
||||||
|
#define PRESERVE_CPU_AFFINITY
|
||||||
|
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
cpuset_t cpuset;
|
cpuset_t cpuset;
|
||||||
|
@ -335,6 +450,28 @@ static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
#include <pthread.h>
|
#include <pthread.h>
|
||||||
#include <sched.h>
|
#include <sched.h>
|
||||||
|
|
||||||
|
INTERNAL_SCOPE cpuset_t *saved_affinity = NULL;
|
||||||
|
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
if (!saved_affinity)
|
||||||
|
saved_affinity = cpuset_create();
|
||||||
|
|
||||||
|
return pthread_getaffinity_np(pthread_self(), cpuset_size(saved_affinity), saved_affinity) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
if (!saved_affinity)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
int ret = pthread_setaffinity_np(pthread_self(), cpuset_size(saved_affinity), saved_affinity) == 0;
|
||||||
|
cpuset_destroy(saved_affinity);
|
||||||
|
saved_affinity = NULL;
|
||||||
|
return ret == 0;
|
||||||
|
}
|
||||||
|
#define PRESERVE_CPU_AFFINITY
|
||||||
|
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
cpuset_t *cpuset;
|
cpuset_t *cpuset;
|
||||||
|
@ -361,6 +498,18 @@ static int get_total_cpus(void)
|
||||||
}
|
}
|
||||||
#endif /* GET_TOTAL_CPUS_DEFINED */
|
#endif /* GET_TOTAL_CPUS_DEFINED */
|
||||||
|
|
||||||
|
#ifndef PRESERVE_CPU_AFFINITY
|
||||||
|
static bool save_cpu_affinity()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool restore_cpu_affinity()
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
#endif /* PRESERVE_CPU_AFFINITY */
|
||||||
|
|
||||||
#ifndef SET_CPU_AFFINITY
|
#ifndef SET_CPU_AFFINITY
|
||||||
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
static bool set_cpu_affinity(logical_cpu_t logical_cpu)
|
||||||
{
|
{
|
||||||
|
@ -924,6 +1073,8 @@ int cpuid_get_all_raw_data(struct cpu_raw_data_array_t* data)
|
||||||
if (data == NULL)
|
if (data == NULL)
|
||||||
return set_error(ERR_HANDLE);
|
return set_error(ERR_HANDLE);
|
||||||
|
|
||||||
|
bool affinity_saved = save_cpu_affinity();
|
||||||
|
|
||||||
cpu_raw_data_array_t_constructor(data, true);
|
cpu_raw_data_array_t_constructor(data, true);
|
||||||
while (set_cpu_affinity(logical_cpu)) {
|
while (set_cpu_affinity(logical_cpu)) {
|
||||||
debugf(2, "Getting raw dump for logical CPU %i\n", logical_cpu);
|
debugf(2, "Getting raw dump for logical CPU %i\n", logical_cpu);
|
||||||
|
@ -935,6 +1086,9 @@ int cpuid_get_all_raw_data(struct cpu_raw_data_array_t* data)
|
||||||
logical_cpu++;
|
logical_cpu++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (affinity_saved)
|
||||||
|
restore_cpu_affinity();
|
||||||
|
|
||||||
return ret_error;
|
return ret_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue