/* * sched_setaffinity.c * * Description: * POSIX scheduling functions that deal with CPU affinity. * * -------------------------------------------------------------------------- * * Pthreads4w - POSIX Threads Library for Win32 * Copyright(C) 1998 John E. Bossom * Copyright(C) 1999-2018, Pthreads4w contributors * * Homepage: https://sourceforge.net/projects/pthreads4w/ * * The current list of contributors is contained * in the file CONTRIBUTORS included with the source * code distribution. The list can also be seen at the * following World Wide Web location: * https://sourceforge.net/p/pthreads4w/wiki/Contributors/ * * This file is part of Pthreads4w. * * Pthreads4w 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. * * Pthreads4w 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 Pthreads4w. If not, see . * */ #ifdef HAVE_CONFIG_H # include #endif #include "pthread.h" #include "implement.h" #include "sched.h" int sched_setaffinity (pid_t pid, size_t cpusetsize, cpu_set_t *set) /* * ------------------------------------------------------ * DOCPUBLIC * Sets the CPU affinity mask of the process whose ID is pid * to the value specified by mask. If pid is zero, then the * calling process is used. The argument cpusetsize is the * length (in bytes) of the data pointed to by mask. Normally * this argument would be specified as sizeof(cpu_set_t). * * If the process specified by pid is not currently running on * one of the CPUs specified in mask, then that process is * migrated to one of the CPUs specified in mask. * * PARAMETERS * pid * Process ID * * cpusetsize * Currently ignored in pthreads4w. * Usually set to sizeof(cpu_set_t) * * mask * Pointer to the CPU mask to set (cpu_set_t). * * DESCRIPTION * Sets the CPU affinity mask of the process whose ID is pid * to the value specified by mask. If pid is zero, then the * calling process is used. The argument cpusetsize is the * length (in bytes) of the data pointed to by mask. Normally * this argument would be specified as sizeof(cpu_set_t). * * If the process specified by pid is not currently running on * one of the CPUs specified in mask, then that process is * migrated to one of the CPUs specified in mask. * * RESULTS * 0 successfully created semaphore, * EFAULT 'mask' is a NULL pointer. * EINVAL '*mask' contains no CPUs in the set * of available CPUs. * EAGAIN The system available CPUs could not * be obtained. * EPERM The process referred to by 'pid' is * not modifiable by us. * ESRCH The process referred to by 'pid' was * not found. * ENOSYS Function not supported. * * ------------------------------------------------------ */ { #if ! defined(NEED_PROCESS_AFFINITY_MASK) DWORD_PTR vProcessMask; DWORD_PTR vSystemMask; HANDLE h; int targetPid = (int)(size_t) pid; int result = 0; if (NULL == set) { result = EFAULT; } else { if (0 == targetPid) { targetPid = (int) GetCurrentProcessId (); } h = OpenProcess (PROCESS_QUERY_INFORMATION|PROCESS_SET_INFORMATION, PTW32_FALSE, (DWORD) targetPid); if (NULL == h) { result = (((0xFF & ERROR_ACCESS_DENIED) == GetLastError()) ? EPERM : ESRCH); } else { if (GetProcessAffinityMask (h, &vProcessMask, &vSystemMask)) { /* * Result is the intersection of available CPUs and the mask. */ DWORD_PTR newMask = vSystemMask & ((_sched_cpu_set_vector_*)set)->_cpuset; if (newMask) { if (SetProcessAffinityMask(h, newMask) == 0) { switch (GetLastError()) { case (0xFF & ERROR_ACCESS_DENIED): result = EPERM; break; case (0xFF & ERROR_INVALID_PARAMETER): result = EINVAL; break; default: result = EAGAIN; break; } } } else { /* * Mask does not contain any CPUs currently available on the system. */ result = EINVAL; } } else { result = EAGAIN; } } CloseHandle(h); } if (result != 0) { PTW32_SET_ERRNO(result); return -1; } else { return 0; } #else PTW32_SET_ERRNO(ENOSYS); return -1; #endif } int sched_getaffinity (pid_t pid, size_t cpusetsize, cpu_set_t *set) /* * ------------------------------------------------------ * DOCPUBLIC * Gets the CPU affinity mask of the process whose ID is pid * to the value specified by mask. If pid is zero, then the * calling process is used. The argument cpusetsize is the * length (in bytes) of the data pointed to by mask. Normally * this argument would be specified as sizeof(cpu_set_t). * * PARAMETERS * pid * Process ID * * cpusetsize * Currently ignored in pthreads4w. * Usually set to sizeof(cpu_set_t) * * mask * Pointer to the CPU mask to set (cpu_set_t). * * DESCRIPTION * Sets the CPU affinity mask of the process whose ID is pid * to the value specified by mask. If pid is zero, then the * calling process is used. The argument cpusetsize is the * length (in bytes) of the data pointed to by mask. Normally * this argument would be specified as sizeof(cpu_set_t). * * RESULTS * 0 successfully created semaphore, * EFAULT 'mask' is a NULL pointer. * EAGAIN The system available CPUs could not * be obtained. * EPERM The process referred to by 'pid' is * not modifiable by us. * ESRCH The process referred to by 'pid' was * not found. * * ------------------------------------------------------ */ { DWORD_PTR vProcessMask; DWORD_PTR vSystemMask; HANDLE h; int targetPid = (int)(size_t) pid; int result = 0; if (NULL == set) { result = EFAULT; } else { #if ! defined(NEED_PROCESS_AFFINITY_MASK) if (0 == targetPid) { targetPid = (int) GetCurrentProcessId (); } h = OpenProcess (PROCESS_QUERY_INFORMATION, PTW32_FALSE, (DWORD) targetPid); if (NULL == h) { result = (((0xFF & ERROR_ACCESS_DENIED) == GetLastError()) ? EPERM : ESRCH); } else { if (GetProcessAffinityMask (h, &vProcessMask, &vSystemMask)) { ((_sched_cpu_set_vector_*)set)->_cpuset = vProcessMask; } else { result = EAGAIN; } } CloseHandle(h); #else ((_sched_cpu_set_vector_*)set)->_cpuset = (size_t)0x1; #endif } if (result != 0) { PTW32_SET_ERRNO(result); return -1; } else { return 0; } } /* * Support routines for cpu_set_t */ int _sched_affinitycpucount (const cpu_set_t *set) { size_t tset; int count; /* * Relies on tset being unsigned, otherwise the right-shift will * be arithmetic rather than logical and the 'for' will loop forever. */ for (count = 0, tset = ((_sched_cpu_set_vector_*)set)->_cpuset; tset; tset >>= 1) { if (tset & (size_t)1) { count++; } } return count; } void _sched_affinitycpuzero (cpu_set_t *pset) { ((_sched_cpu_set_vector_*)pset)->_cpuset = (size_t)0; } void _sched_affinitycpuset (int cpu, cpu_set_t *pset) { ((_sched_cpu_set_vector_*)pset)->_cpuset |= ((size_t)1 << cpu); } void _sched_affinitycpuclr (int cpu, cpu_set_t *pset) { ((_sched_cpu_set_vector_*)pset)->_cpuset &= ~((size_t)1 << cpu); } int _sched_affinitycpuisset (int cpu, const cpu_set_t *pset) { return ((((_sched_cpu_set_vector_*)pset)->_cpuset & ((size_t)1 << cpu)) != (size_t)0); } void _sched_affinitycpuand(cpu_set_t *pdestset, const cpu_set_t *psrcset1, const cpu_set_t *psrcset2) { ((_sched_cpu_set_vector_*)pdestset)->_cpuset = (((_sched_cpu_set_vector_*)psrcset1)->_cpuset & ((_sched_cpu_set_vector_*)psrcset2)->_cpuset); } void _sched_affinitycpuor(cpu_set_t *pdestset, const cpu_set_t *psrcset1, const cpu_set_t *psrcset2) { ((_sched_cpu_set_vector_*)pdestset)->_cpuset = (((_sched_cpu_set_vector_*)psrcset1)->_cpuset | ((_sched_cpu_set_vector_*)psrcset2)->_cpuset); } void _sched_affinitycpuxor(cpu_set_t *pdestset, const cpu_set_t *psrcset1, const cpu_set_t *psrcset2) { ((_sched_cpu_set_vector_*)pdestset)->_cpuset = (((_sched_cpu_set_vector_*)psrcset1)->_cpuset ^ ((_sched_cpu_set_vector_*)psrcset2)->_cpuset); } int _sched_affinitycpuequal (const cpu_set_t *pset1, const cpu_set_t *pset2) { return (((_sched_cpu_set_vector_*)pset1)->_cpuset == ((_sched_cpu_set_vector_*)pset2)->_cpuset); }