xref: /src/contrib/llvm-project/llvm/lib/Support/Threading.cpp (revision 06c3fb2749bda94cb5201f81ffdb8fa6c3161b2e)
1cf099d11SDimitry Andric //===-- llvm/Support/Threading.cpp- Control multithreading mode --*- C++ -*-==//
2cf099d11SDimitry Andric //
3e6d15924SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e6d15924SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5e6d15924SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6cf099d11SDimitry Andric //
7cf099d11SDimitry Andric //===----------------------------------------------------------------------===//
8cf099d11SDimitry Andric //
95ca98fd9SDimitry Andric // This file defines helper functions for running LLVM in a multi-threaded
105ca98fd9SDimitry Andric // environment.
11cf099d11SDimitry Andric //
12cf099d11SDimitry Andric //===----------------------------------------------------------------------===//
13cf099d11SDimitry Andric 
14cf099d11SDimitry Andric #include "llvm/Support/Threading.h"
154a16efa3SDimitry Andric #include "llvm/Config/config.h"
16bdd1243dSDimitry Andric #include "llvm/Config/llvm-config.h"
1771d5a254SDimitry Andric 
18cf099d11SDimitry Andric #include <cassert>
1971d5a254SDimitry Andric #include <errno.h>
20bdd1243dSDimitry Andric #include <optional>
2171d5a254SDimitry Andric #include <stdlib.h>
2271d5a254SDimitry Andric #include <string.h>
23cf099d11SDimitry Andric 
24cf099d11SDimitry Andric using namespace llvm;
25cf099d11SDimitry Andric 
2671d5a254SDimitry Andric //===----------------------------------------------------------------------===//
2771d5a254SDimitry Andric //=== WARNING: Implementation here must contain only TRULY operating system
2871d5a254SDimitry Andric //===          independent code.
2971d5a254SDimitry Andric //===----------------------------------------------------------------------===//
3071d5a254SDimitry Andric 
3171d5a254SDimitry Andric #if LLVM_ENABLE_THREADS == 0 ||                                                \
32eb11fae6SDimitry Andric     (!defined(_WIN32) && !defined(HAVE_PTHREAD_H))
get_threadid()3371d5a254SDimitry Andric uint64_t llvm::get_threadid() { return 0; }
3471d5a254SDimitry Andric 
get_max_thread_name_length()3571d5a254SDimitry Andric uint32_t llvm::get_max_thread_name_length() { return 0; }
3671d5a254SDimitry Andric 
set_thread_name(const Twine & Name)3771d5a254SDimitry Andric void llvm::set_thread_name(const Twine &Name) {}
3871d5a254SDimitry Andric 
get_thread_name(SmallVectorImpl<char> & Name)3971d5a254SDimitry Andric void llvm::get_thread_name(SmallVectorImpl<char> &Name) { Name.clear(); }
4071d5a254SDimitry Andric 
get_thread_affinity_mask()41cfca06d7SDimitry Andric llvm::BitVector llvm::get_thread_affinity_mask() { return {}; }
42cfca06d7SDimitry Andric 
compute_thread_count() const43cfca06d7SDimitry Andric unsigned llvm::ThreadPoolStrategy::compute_thread_count() const {
44cfca06d7SDimitry Andric   // When threads are disabled, ensure clients will loop at least once.
45cfca06d7SDimitry Andric   return 1;
46cfca06d7SDimitry Andric }
47cfca06d7SDimitry Andric 
48bdd1243dSDimitry Andric // Unknown if threading turned off
get_physical_cores()49bdd1243dSDimitry Andric int llvm::get_physical_cores() { return -1; }
50bdd1243dSDimitry Andric 
5171d5a254SDimitry Andric #else
5271d5a254SDimitry Andric 
53bdd1243dSDimitry Andric static int computeHostNumHardwareThreads();
5471d5a254SDimitry Andric 
compute_thread_count() const55cfca06d7SDimitry Andric unsigned llvm::ThreadPoolStrategy::compute_thread_count() const {
56bdd1243dSDimitry Andric   int MaxThreadCount =
57bdd1243dSDimitry Andric       UseHyperThreads ? computeHostNumHardwareThreads() : get_physical_cores();
58cfca06d7SDimitry Andric   if (MaxThreadCount <= 0)
59cfca06d7SDimitry Andric     MaxThreadCount = 1;
6004bab189SMateusz Guzik   // Damage control threading.
6104bab189SMateusz Guzik   //
6204bab189SMateusz Guzik   // There are no heuristics to figure out how many threads makes sense to spawn,
6304bab189SMateusz Guzik   // all while rolling with all available hw threads starts being detrimental to
6404bab189SMateusz Guzik   // performance really early.
6504bab189SMateusz Guzik   //
6604bab189SMateusz Guzik   // Work around by putting a hard cap unless the user explicitly requested a certain amount.
6704bab189SMateusz Guzik   //
6804bab189SMateusz Guzik   // See https://discourse.llvm.org/t/avoidable-overhead-from-threading-by-default/69160
6904bab189SMateusz Guzik   // for more details.
7004bab189SMateusz Guzik   if (ThreadsRequested == 0) {
7104bab189SMateusz Guzik     return std::min(MaxThreadCount, 4);
7204bab189SMateusz Guzik   }
73cfca06d7SDimitry Andric   if (!Limit)
74cfca06d7SDimitry Andric     return ThreadsRequested;
75cfca06d7SDimitry Andric   return std::min((unsigned)MaxThreadCount, ThreadsRequested);
76044eb2f6SDimitry Andric }
77044eb2f6SDimitry Andric 
7871d5a254SDimitry Andric // Include the platform-specific parts of this class.
7971d5a254SDimitry Andric #ifdef LLVM_ON_UNIX
8071d5a254SDimitry Andric #include "Unix/Threading.inc"
8171d5a254SDimitry Andric #endif
82eb11fae6SDimitry Andric #ifdef _WIN32
8371d5a254SDimitry Andric #include "Windows/Threading.inc"
8471d5a254SDimitry Andric #endif
8571d5a254SDimitry Andric 
86344a3780SDimitry Andric // Must be included after Threading.inc to provide definition for llvm::thread
87344a3780SDimitry Andric // because FreeBSD's condvar.h (included by user.h) misuses the "thread"
88344a3780SDimitry Andric // keyword.
89344a3780SDimitry Andric #include "llvm/Support/thread.h"
90706b4fc4SDimitry Andric 
91344a3780SDimitry Andric #if defined(__APPLE__)
92344a3780SDimitry Andric   // Darwin's default stack size for threads except the main one is only 512KB,
93344a3780SDimitry Andric   // which is not enough for some/many normal LLVM compilations. This implements
94344a3780SDimitry Andric   // the same interface as std::thread but requests the same stack size as the
95344a3780SDimitry Andric   // main thread (8MB) before creation.
96bdd1243dSDimitry Andric const std::optional<unsigned> llvm::thread::DefaultStackSize = 8 * 1024 * 1024;
97*06c3fb27SDimitry Andric #elif defined(_AIX)
98*06c3fb27SDimitry Andric   // On AIX, the default pthread stack size limit is ~192k for 64-bit programs.
99*06c3fb27SDimitry Andric   // This limit is easily reached when doing link-time thinLTO. AIX library
100*06c3fb27SDimitry Andric   // developers have used 4MB, so we'll do the same.
101*06c3fb27SDimitry Andric const std::optional<unsigned> llvm::thread::DefaultStackSize = 4 * 1024 * 1024;
102344a3780SDimitry Andric #else
103bdd1243dSDimitry Andric const std::optional<unsigned> llvm::thread::DefaultStackSize;
104344a3780SDimitry Andric #endif
105706b4fc4SDimitry Andric 
106706b4fc4SDimitry Andric 
10771d5a254SDimitry Andric #endif
108cfca06d7SDimitry Andric 
109bdd1243dSDimitry Andric std::optional<ThreadPoolStrategy>
get_threadpool_strategy(StringRef Num,ThreadPoolStrategy Default)110cfca06d7SDimitry Andric llvm::get_threadpool_strategy(StringRef Num, ThreadPoolStrategy Default) {
111cfca06d7SDimitry Andric   if (Num == "all")
112cfca06d7SDimitry Andric     return llvm::hardware_concurrency();
113cfca06d7SDimitry Andric   if (Num.empty())
114cfca06d7SDimitry Andric     return Default;
115cfca06d7SDimitry Andric   unsigned V;
116cfca06d7SDimitry Andric   if (Num.getAsInteger(10, V))
117bdd1243dSDimitry Andric     return std::nullopt; // malformed 'Num' value
118cfca06d7SDimitry Andric   if (V == 0)
119cfca06d7SDimitry Andric     return Default;
120cfca06d7SDimitry Andric 
121cfca06d7SDimitry Andric   // Do not take the Default into account. This effectively disables
122cfca06d7SDimitry Andric   // heavyweight_hardware_concurrency() if the user asks for any number of
123cfca06d7SDimitry Andric   // threads on the cmd-line.
124cfca06d7SDimitry Andric   ThreadPoolStrategy S = llvm::hardware_concurrency();
125cfca06d7SDimitry Andric   S.ThreadsRequested = V;
126cfca06d7SDimitry Andric   return S;
127cfca06d7SDimitry Andric }
128