23#include <mach/mach_init.h>
24#include <mach/mach_port.h>
25#include <pthread/qos.h>
26#include <sys/sysctl.h>
32#if defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__DragonFly__)
33#include <pthread_np.h>
38#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
40#include <sys/cpuset.h>
41#include <sys/sysctl.h>
46#if defined(__NetBSD__)
50#if defined(__OpenBSD__)
56#include <sys/syscall.h>
60#if defined(__CYGWIN__)
61#include <sys/cpuset.h>
70llvm_execute_on_thread_impl(
void *(*ThreadFunc)(
void *),
void *Arg,
71 std::optional<unsigned> StackSizeInBytes) {
76 if ((errnum = ::pthread_attr_init(&Attr)) != 0) {
81 if ((errnum = ::pthread_attr_destroy(&Attr)) != 0) {
87 if (StackSizeInBytes) {
88 if ((errnum = ::pthread_attr_setstacksize(&Attr, *StackSizeInBytes)) != 0) {
95 if ((errnum = ::pthread_create(&Thread, &Attr, ThreadFunc, Arg)) != 0)
101void llvm_thread_detach_impl(pthread_t Thread) {
104 if ((errnum = ::pthread_detach(Thread)) != 0) {
109void llvm_thread_join_impl(pthread_t Thread) {
112 if ((errnum = ::pthread_join(Thread,
nullptr)) != 0) {
117llvm::thread::id llvm_thread_get_id_impl(pthread_t Thread) {
125llvm::thread::id llvm_thread_get_current_id_impl() {
126 return llvm_thread_get_id_impl(::pthread_self());
132#if defined(__APPLE__)
136 static thread_local thread_port_t Self = [] {
137 thread_port_t InitSelf = mach_thread_self();
138 mach_port_deallocate(mach_task_self(), Self);
142#elif defined(__FreeBSD__) || defined(__DragonFly__)
143 return uint64_t(pthread_getthreadid_np());
144#elif defined(__NetBSD__)
146#elif defined(__OpenBSD__)
148#elif defined(__ANDROID__)
150#elif defined(__linux__)
151 return uint64_t(syscall(__NR_gettid));
154#elif defined(__MVS__)
155 return llvm_thread_get_id_impl(pthread_self());
161static constexpr uint32_t get_max_thread_name_length_impl() {
162#if defined(PTHREAD_MAX_NAMELEN_NP)
163 return PTHREAD_MAX_NAMELEN_NP;
164#elif defined(__HAIKU__)
165 return B_OS_NAME_LENGTH;
166#elif defined(__APPLE__)
168#elif defined(__sun__) && defined(__svr4__)
170#elif defined(__linux__) && HAVE_PTHREAD_SETNAME_NP
172#elif defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
173 defined(__DragonFly__)
175#elif defined(__OpenBSD__)
177#elif defined(__CYGWIN__)
185 return get_max_thread_name_length_impl();
190 SmallString<64> Storage;
191 StringRef NameStr =
Name.toNullTerminatedStringRef(Storage);
202#if defined(HAVE_PTHREAD_SET_NAME_NP) && HAVE_PTHREAD_SET_NAME_NP
203 ::pthread_set_name_np(::pthread_self(), NameStr.data());
204#elif defined(HAVE_PTHREAD_SETNAME_NP) && HAVE_PTHREAD_SETNAME_NP
205#if defined(__NetBSD__)
206 ::pthread_setname_np(::pthread_self(),
"%s",
207 const_cast<char *
>(NameStr.data()));
208#elif defined(__APPLE__)
209 ::pthread_setname_np(NameStr.data());
211 ::pthread_setname_np(::pthread_self(), NameStr.data());
219#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
220 int pid = ::getpid();
223 struct kinfo_proc *kp =
nullptr, *nkp;
226 int ctl[4] = {CTL_KERN, KERN_PROC, KERN_PROC_PID | KERN_PROC_INC_THREAD,
230 error = sysctl(ctl, 4, kp, &len,
nullptr, 0);
231 if (kp ==
nullptr || (
error != 0 && errno == ENOMEM)) {
233 len +=
sizeof(*kp) + len / 10;
234 nkp = (
struct kinfo_proc *)::realloc(kp, len);
235 if (nkp ==
nullptr) {
247 for (
size_t i = 0; i < len /
sizeof(*kp); i++) {
248 if (kp[i].ki_tid == (lwpid_t)tid) {
249 Name.append(kp[i].ki_tdname, kp[i].ki_tdname + strlen(kp[i].ki_tdname));
255#elif (defined(__linux__) || defined(__CYGWIN__)) && HAVE_PTHREAD_GETNAME_NP
256 constexpr uint32_t len = get_max_thread_name_length_impl();
257 char Buffer[len] = {
'\0'};
258 if (0 == ::pthread_getname_np(::pthread_self(), Buffer, len))
259 Name.append(Buffer, Buffer + strlen(Buffer));
260#elif defined(HAVE_PTHREAD_GET_NAME_NP) && HAVE_PTHREAD_GET_NAME_NP
261 constexpr uint32_t len = get_max_thread_name_length_impl();
263 ::pthread_get_name_np(::pthread_self(), buf, len);
265 Name.append(buf, buf + strlen(buf));
267#elif defined(HAVE_PTHREAD_GETNAME_NP) && HAVE_PTHREAD_GETNAME_NP
268 constexpr uint32_t len = get_max_thread_name_length_impl();
270 ::pthread_getname_np(::pthread_self(), buf, len);
272 Name.append(buf, buf + strlen(buf));
278#if (defined(__linux__) || defined(__CYGWIN__)) && defined(SCHED_IDLE)
282 sched_param priority;
284 priority.sched_priority = 0;
287 return !pthread_setschedparam(
290 Priority == ThreadPriority::Default ? SCHED_OTHER : SCHED_IDLE,
292 ? SetThreadPriorityResult::SUCCESS
293 : SetThreadPriorityResult::FAILURE;
294#elif defined(__APPLE__)
305 const auto qosClass = [&]() {
307 case ThreadPriority::Background:
308 return QOS_CLASS_BACKGROUND;
309 case ThreadPriority::Low:
310 return QOS_CLASS_UTILITY;
311 case ThreadPriority::Default:
312 return QOS_CLASS_DEFAULT;
315 return !pthread_set_qos_class_self_np(qosClass, 0)
316 ? SetThreadPriorityResult::SUCCESS
317 : SetThreadPriorityResult::FAILURE;
319 return SetThreadPriorityResult::FAILURE;
324static int computeHostNumHardwareThreads() {
325#if defined(__FreeBSD__)
328 if (cpuset_getaffinity(CPU_LEVEL_WHICH, CPU_WHICH_TID, -1,
sizeof(
mask),
330 return CPU_COUNT(&
mask);
331#elif (defined(__linux__) || defined(__CYGWIN__))
334 if (sched_getaffinity(0,
sizeof(Set), &Set) == 0)
335 return CPU_COUNT(&Set);
338 if (
unsigned Val = std::thread::hardware_concurrency())
344 unsigned ThreadPoolNum)
const {}
353#if (defined(__linux__) || defined(__CYGWIN__)) && \
354 (defined(__i386__) || defined(__x86_64__))
358static int computeHostNumPhysicalCores() {
362 if (sched_getaffinity(0,
sizeof(Affinity), &Affinity) != 0)
370 if (std::error_code EC =
Text.getError()) {
372 <<
"/proc/cpuinfo: " <<
EC.message() <<
"\n";
376 (*Text)->getBuffer().split(strs,
"\n", -1,
378 int CurProcessor = -1;
379 int CurPhysicalId = -1;
380 int CurSiblings = -1;
383 std::pair<llvm::StringRef, llvm::StringRef>
Data =
Line.split(
':');
385 auto Val =
Data.second.trim();
387 if (Name ==
"processor")
388 Val.getAsInteger(10, CurProcessor);
389 else if (Name ==
"physical id")
390 Val.getAsInteger(10, CurPhysicalId);
391 else if (Name ==
"siblings")
392 Val.getAsInteger(10, CurSiblings);
393 else if (Name ==
"core id") {
394 Val.getAsInteger(10, CurCoreId);
396 if (CPU_ISSET(CurProcessor, &Affinity))
397 CPU_SET(CurPhysicalId * CurSiblings + CurCoreId, &
Enabled);
402#elif (defined(__linux__) && defined(__s390x__)) || defined(_AIX)
403static int computeHostNumPhysicalCores() {
404 return sysconf(_SC_NPROCESSORS_ONLN);
406#elif defined(__linux__)
407static int computeHostNumPhysicalCores() {
409 if (sched_getaffinity(0,
sizeof(Affinity), &Affinity) == 0)
410 return CPU_COUNT(&Affinity);
416 cpu_set_t *DynAffinity;
417 DynAffinity = CPU_ALLOC(2048);
418 if (sched_getaffinity(0, CPU_ALLOC_SIZE(2048), DynAffinity) == 0) {
419 int NumCPUs = CPU_COUNT(DynAffinity);
420 CPU_FREE(DynAffinity);
425#elif defined(__APPLE__)
427static int computeHostNumPhysicalCores() {
429 size_t len =
sizeof(
count);
430 sysctlbyname(
"hw.physicalcpu", &
count, &len, NULL, 0);
435 sysctl(nm, 2, &
count, &len, NULL, 0);
441#elif defined(__MVS__)
442static int computeHostNumPhysicalCores() {
454 CSD_NUMBER_ONLINE_STANDARD_CPS = 264,
457 char *CVT =
reinterpret_cast<char *
>(
458 static_cast<uintptr_t
>(
reinterpret_cast<unsigned int &
>(PSA[FLCCVT])));
459 char *CSD =
reinterpret_cast<char *
>(
460 static_cast<uintptr_t
>(
reinterpret_cast<unsigned int &
>(CVT[CVTCSD])));
461 return reinterpret_cast<int &
>(CSD[CSD_NUMBER_ONLINE_STANDARD_CPS]);
465static int computeHostNumPhysicalCores() {
return -1; }
469 static int NumCores = computeHostNumPhysicalCores();
static constexpr unsigned long long mask(BlockVerifier::State S)
This file defines the make_scope_exit function, which executes user-defined cleanup logic at scope ex...
This file defines the SmallString class.
This file defines the SmallVector class.
static void ReportErrnumFatal(const char *Msg, int errnum)
Represents either an error or a value T.
static ErrorOr< std::unique_ptr< MemoryBuffer > > getFileAsStream(const Twine &Filename)
Read all of the specified file into a MemoryBuffer as a stream (i.e.
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringRef - Represent a constant reference to a string, i.e.
LLVM_ABI void apply_thread_strategy(unsigned ThreadPoolNum) const
Assign the current thread to an ideal hardware CPU or NUMA node.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
This is an optimization pass for GlobalISel generic memory operations.
LLVM_ABI llvm::BitVector get_thread_affinity_mask()
Returns a mask that represents on which hardware thread, core, CPU, NUMA group, the calling thread ca...
LLVM_ABI uint32_t get_max_thread_name_length()
Get the maximum length of a thread name on this platform.
LLVM_ABI SetThreadPriorityResult set_thread_priority(ThreadPriority Priority)
LLVM_ABI unsigned get_cpus()
Returns how many physical CPUs or NUMA groups the system has.
LLVM_ABI void set_thread_name(const Twine &Name)
Set the name of the current thread.
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
LLVM_ABI void get_thread_name(SmallVectorImpl< char > &Name)
Get the name of the current thread.
FunctionAddr VTableAddr uintptr_t uintptr_t Data
LLVM_ABI int get_physical_cores()
Returns how many physical cores (as opposed to logical cores returned from thread::hardware_concurren...
LLVM_ABI uint64_t get_threadid()
Return the current thread id, as used in various OS system calls.
auto count(R &&Range, const E &Element)
Wrapper function around std::count to count the number of times an element Element occurs in the give...
@ Enabled
Convert any .debug_str_offsets tables to DWARF64 if needed.