| // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "base/threading/platform_thread.h" | 
 |  | 
 | #include <errno.h> | 
 | #include <sched.h> | 
 |  | 
 | #include "base/files/file_util.h" | 
 | #include "base/lazy_instance.h" | 
 | #include "base/logging.h" | 
 | #include "base/strings/string_number_conversions.h" | 
 | #include "base/threading/platform_thread_internal_posix.h" | 
 | #include "base/threading/thread_id_name_manager.h" | 
 | #include "build/build_config.h" | 
 |  | 
 | #if !defined(OS_NACL) && !defined(OS_AIX) | 
 | #include <pthread.h> | 
 | #include <sys/prctl.h> | 
 | #include <sys/resource.h> | 
 | #include <sys/time.h> | 
 | #include <sys/types.h> | 
 | #include <unistd.h> | 
 |  | 
 | #include "starboard/types.h" | 
 | #endif | 
 |  | 
 | namespace base { | 
 | namespace { | 
 | #if !defined(OS_NACL) | 
 | const FilePath::CharType kCgroupDirectory[] = | 
 |     FILE_PATH_LITERAL("/sys/fs/cgroup"); | 
 |  | 
 | FilePath ThreadPriorityToCgroupDirectory(const FilePath& cgroup_filepath, | 
 |                                          ThreadPriority priority) { | 
 |   switch (priority) { | 
 |     case ThreadPriority::NORMAL: | 
 |       return cgroup_filepath; | 
 |     case ThreadPriority::BACKGROUND: | 
 |       return cgroup_filepath.Append(FILE_PATH_LITERAL("non-urgent")); | 
 |     case ThreadPriority::DISPLAY: | 
 |     case ThreadPriority::REALTIME_AUDIO: | 
 |       return cgroup_filepath.Append(FILE_PATH_LITERAL("urgent")); | 
 |   } | 
 |   NOTREACHED(); | 
 |   return FilePath(); | 
 | } | 
 |  | 
 | void SetThreadCgroup(PlatformThreadId thread_id, | 
 |                      const FilePath& cgroup_directory) { | 
 |   FilePath tasks_filepath = cgroup_directory.Append(FILE_PATH_LITERAL("tasks")); | 
 |   std::string tid = IntToString(thread_id); | 
 |   int bytes_written = WriteFile(tasks_filepath, tid.c_str(), tid.size()); | 
 |   if (bytes_written != static_cast<int>(tid.size())) { | 
 |     DVLOG(1) << "Failed to add " << tid << " to " << tasks_filepath.value(); | 
 |   } | 
 | } | 
 |  | 
 | void SetThreadCgroupForThreadPriority(PlatformThreadId thread_id, | 
 |                                       const FilePath& cgroup_filepath, | 
 |                                       ThreadPriority priority) { | 
 |   // Append "chrome" suffix. | 
 |   FilePath cgroup_directory = ThreadPriorityToCgroupDirectory( | 
 |       cgroup_filepath.Append(FILE_PATH_LITERAL("chrome")), priority); | 
 |  | 
 |   // Silently ignore request if cgroup directory doesn't exist. | 
 |   if (!DirectoryExists(cgroup_directory)) | 
 |     return; | 
 |  | 
 |   SetThreadCgroup(thread_id, cgroup_directory); | 
 | } | 
 |  | 
 | void SetThreadCgroupsForThreadPriority(PlatformThreadId thread_id, | 
 |                                        ThreadPriority priority) { | 
 |   FilePath cgroup_filepath(kCgroupDirectory); | 
 |   SetThreadCgroupForThreadPriority( | 
 |       thread_id, cgroup_filepath.Append(FILE_PATH_LITERAL("cpuset")), priority); | 
 |   SetThreadCgroupForThreadPriority( | 
 |       thread_id, cgroup_filepath.Append(FILE_PATH_LITERAL("schedtune")), | 
 |       priority); | 
 | } | 
 | #endif | 
 | }  // namespace | 
 |  | 
 | namespace internal { | 
 |  | 
 | namespace { | 
 | #if !defined(OS_NACL) | 
 | const struct sched_param kRealTimePrio = {8}; | 
 | #endif | 
 | }  // namespace | 
 |  | 
 | const ThreadPriorityToNiceValuePair kThreadPriorityToNiceValueMap[4] = { | 
 |     {ThreadPriority::BACKGROUND, 10}, | 
 |     {ThreadPriority::NORMAL, 0}, | 
 |     {ThreadPriority::DISPLAY, -8}, | 
 |     {ThreadPriority::REALTIME_AUDIO, -10}, | 
 | }; | 
 |  | 
 | bool SetCurrentThreadPriorityForPlatform(ThreadPriority priority) { | 
 | #if !defined(OS_NACL) | 
 |   SetThreadCgroupsForThreadPriority(PlatformThread::CurrentId(), priority); | 
 |   return priority == ThreadPriority::REALTIME_AUDIO && | 
 |          pthread_setschedparam(pthread_self(), SCHED_RR, &kRealTimePrio) == 0; | 
 | #else | 
 |   return false; | 
 | #endif | 
 | } | 
 |  | 
 | bool GetCurrentThreadPriorityForPlatform(ThreadPriority* priority) { | 
 | #if !defined(OS_NACL) | 
 |   int maybe_sched_rr = 0; | 
 |   struct sched_param maybe_realtime_prio = {0}; | 
 |   if (pthread_getschedparam(pthread_self(), &maybe_sched_rr, | 
 |                             &maybe_realtime_prio) == 0 && | 
 |       maybe_sched_rr == SCHED_RR && | 
 |       maybe_realtime_prio.sched_priority == kRealTimePrio.sched_priority) { | 
 |     *priority = ThreadPriority::REALTIME_AUDIO; | 
 |     return true; | 
 |   } | 
 | #endif | 
 |   return false; | 
 | } | 
 |  | 
 | }  // namespace internal | 
 |  | 
 | // static | 
 | void PlatformThread::SetName(const std::string& name) { | 
 |   ThreadIdNameManager::GetInstance()->SetName(name); | 
 |  | 
 | #if !defined(OS_NACL) && !defined(OS_AIX) | 
 |   // On linux we can get the thread names to show up in the debugger by setting | 
 |   // the process name for the LWP.  We don't want to do this for the main | 
 |   // thread because that would rename the process, causing tools like killall | 
 |   // to stop working. | 
 |   if (PlatformThread::CurrentId() == getpid()) | 
 |     return; | 
 |  | 
 |   // http://0pointer.de/blog/projects/name-your-threads.html | 
 |   // Set the name for the LWP (which gets truncated to 15 characters). | 
 |   // Note that glibc also has a 'pthread_setname_np' api, but it may not be | 
 |   // available everywhere and it's only benefit over using prctl directly is | 
 |   // that it can set the name of threads other than the current thread. | 
 |   int err = prctl(PR_SET_NAME, name.c_str()); | 
 |   // We expect EPERM failures in sandboxed processes, just ignore those. | 
 |   if (err < 0 && errno != EPERM) | 
 |     DPLOG(ERROR) << "prctl(PR_SET_NAME)"; | 
 | #endif  //  !defined(OS_NACL) && !defined(OS_AIX) | 
 | } | 
 |  | 
 | #if !defined(OS_NACL) && !defined(OS_AIX) | 
 | // static | 
 | void PlatformThread::SetThreadPriority(PlatformThreadId thread_id, | 
 |                                        ThreadPriority priority) { | 
 |   // Changing current main threads' priority is not permitted in favor of | 
 |   // security, this interface is restricted to change only non-main thread | 
 |   // priority. | 
 |   CHECK_NE(thread_id, getpid()); | 
 |  | 
 |   SetThreadCgroupsForThreadPriority(thread_id, priority); | 
 |  | 
 |   const int nice_setting = internal::ThreadPriorityToNiceValue(priority); | 
 |   if (setpriority(PRIO_PROCESS, thread_id, nice_setting)) { | 
 |     DVPLOG(1) << "Failed to set nice value of thread (" << thread_id << ") to " | 
 |               << nice_setting; | 
 |   } | 
 | } | 
 | #endif  //  !defined(OS_NACL) && !defined(OS_AIX) | 
 |  | 
 | void InitThreading() {} | 
 |  | 
 | void TerminateOnThread() {} | 
 |  | 
 | size_t GetDefaultThreadStackSize(const pthread_attr_t& attributes) { | 
 | #if !defined(THREAD_SANITIZER) | 
 |   return 0; | 
 | #else | 
 |   // ThreadSanitizer bloats the stack heavily. Evidence has been that the | 
 |   // default stack size isn't enough for some browser tests. | 
 |   return 2 * (1 << 23);  // 2 times 8192K (the default stack size on Linux). | 
 | #endif | 
 | } | 
 |  | 
 | }  // namespace base |