xref: /aosp_15_r20/external/cronet/base/process/process_win.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2011 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "base/process/process.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <windows.h>
8*6777b538SAndroid Build Coastguard Worker 
9*6777b538SAndroid Build Coastguard Worker #include "base/clang_profiling_buildflags.h"
10*6777b538SAndroid Build Coastguard Worker #include "base/logging.h"
11*6777b538SAndroid Build Coastguard Worker #include "base/numerics/safe_conversions.h"
12*6777b538SAndroid Build Coastguard Worker #include "base/process/kill.h"
13*6777b538SAndroid Build Coastguard Worker #include "base/threading/thread_restrictions.h"
14*6777b538SAndroid Build Coastguard Worker #include "base/trace_event/base_tracing.h"
15*6777b538SAndroid Build Coastguard Worker #include "base/win/windows_version.h"
16*6777b538SAndroid Build Coastguard Worker 
17*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(CLANG_PROFILING)
18*6777b538SAndroid Build Coastguard Worker #include "base/test/clang_profiling.h"
19*6777b538SAndroid Build Coastguard Worker #endif
20*6777b538SAndroid Build Coastguard Worker 
21*6777b538SAndroid Build Coastguard Worker namespace {
22*6777b538SAndroid Build Coastguard Worker 
23*6777b538SAndroid Build Coastguard Worker DWORD kBasicProcessAccess =
24*6777b538SAndroid Build Coastguard Worker   PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION | SYNCHRONIZE;
25*6777b538SAndroid Build Coastguard Worker 
26*6777b538SAndroid Build Coastguard Worker } // namespace
27*6777b538SAndroid Build Coastguard Worker 
28*6777b538SAndroid Build Coastguard Worker namespace base {
29*6777b538SAndroid Build Coastguard Worker 
30*6777b538SAndroid Build Coastguard Worker // Sets Eco QoS (Quality of Service) level for background process which would
31*6777b538SAndroid Build Coastguard Worker // select efficient CPU frequency and schedule the process to efficient cores
32*6777b538SAndroid Build Coastguard Worker // (available on hybrid CPUs).
33*6777b538SAndroid Build Coastguard Worker // QoS is a scheduling Win API which indicates the desired performance and power
34*6777b538SAndroid Build Coastguard Worker // efficiency of a process/thread. EcoQoS is introduced since Windows 11.
35*6777b538SAndroid Build Coastguard Worker BASE_FEATURE(kUseEcoQoSForBackgroundProcess,
36*6777b538SAndroid Build Coastguard Worker              "UseEcoQoSForBackgroundProcess",
37*6777b538SAndroid Build Coastguard Worker              FEATURE_ENABLED_BY_DEFAULT);
38*6777b538SAndroid Build Coastguard Worker 
Process(ProcessHandle handle)39*6777b538SAndroid Build Coastguard Worker Process::Process(ProcessHandle handle)
40*6777b538SAndroid Build Coastguard Worker     : process_(handle), is_current_process_(false) {
41*6777b538SAndroid Build Coastguard Worker   CHECK_NE(handle, ::GetCurrentProcess());
42*6777b538SAndroid Build Coastguard Worker }
43*6777b538SAndroid Build Coastguard Worker 
Process(Process && other)44*6777b538SAndroid Build Coastguard Worker Process::Process(Process&& other)
45*6777b538SAndroid Build Coastguard Worker     : process_(other.process_.release()),
46*6777b538SAndroid Build Coastguard Worker       is_current_process_(other.is_current_process_) {
47*6777b538SAndroid Build Coastguard Worker   other.Close();
48*6777b538SAndroid Build Coastguard Worker }
49*6777b538SAndroid Build Coastguard Worker 
~Process()50*6777b538SAndroid Build Coastguard Worker Process::~Process() {
51*6777b538SAndroid Build Coastguard Worker }
52*6777b538SAndroid Build Coastguard Worker 
operator =(Process && other)53*6777b538SAndroid Build Coastguard Worker Process& Process::operator=(Process&& other) {
54*6777b538SAndroid Build Coastguard Worker   DCHECK_NE(this, &other);
55*6777b538SAndroid Build Coastguard Worker   process_.Set(other.process_.release());
56*6777b538SAndroid Build Coastguard Worker   is_current_process_ = other.is_current_process_;
57*6777b538SAndroid Build Coastguard Worker   other.Close();
58*6777b538SAndroid Build Coastguard Worker   return *this;
59*6777b538SAndroid Build Coastguard Worker }
60*6777b538SAndroid Build Coastguard Worker 
61*6777b538SAndroid Build Coastguard Worker // static
Current()62*6777b538SAndroid Build Coastguard Worker Process Process::Current() {
63*6777b538SAndroid Build Coastguard Worker   Process process;
64*6777b538SAndroid Build Coastguard Worker   process.is_current_process_ = true;
65*6777b538SAndroid Build Coastguard Worker   return process;
66*6777b538SAndroid Build Coastguard Worker }
67*6777b538SAndroid Build Coastguard Worker 
68*6777b538SAndroid Build Coastguard Worker // static
Open(ProcessId pid)69*6777b538SAndroid Build Coastguard Worker Process Process::Open(ProcessId pid) {
70*6777b538SAndroid Build Coastguard Worker   return Process(::OpenProcess(kBasicProcessAccess, FALSE, pid));
71*6777b538SAndroid Build Coastguard Worker }
72*6777b538SAndroid Build Coastguard Worker 
73*6777b538SAndroid Build Coastguard Worker // static
OpenWithExtraPrivileges(ProcessId pid)74*6777b538SAndroid Build Coastguard Worker Process Process::OpenWithExtraPrivileges(ProcessId pid) {
75*6777b538SAndroid Build Coastguard Worker   DWORD access = kBasicProcessAccess | PROCESS_DUP_HANDLE | PROCESS_VM_READ;
76*6777b538SAndroid Build Coastguard Worker   return Process(::OpenProcess(access, FALSE, pid));
77*6777b538SAndroid Build Coastguard Worker }
78*6777b538SAndroid Build Coastguard Worker 
79*6777b538SAndroid Build Coastguard Worker // static
OpenWithAccess(ProcessId pid,DWORD desired_access)80*6777b538SAndroid Build Coastguard Worker Process Process::OpenWithAccess(ProcessId pid, DWORD desired_access) {
81*6777b538SAndroid Build Coastguard Worker   return Process(::OpenProcess(desired_access, FALSE, pid));
82*6777b538SAndroid Build Coastguard Worker }
83*6777b538SAndroid Build Coastguard Worker 
84*6777b538SAndroid Build Coastguard Worker // static
CanSetPriority()85*6777b538SAndroid Build Coastguard Worker bool Process::CanSetPriority() {
86*6777b538SAndroid Build Coastguard Worker   return true;
87*6777b538SAndroid Build Coastguard Worker }
88*6777b538SAndroid Build Coastguard Worker 
89*6777b538SAndroid Build Coastguard Worker // static
TerminateCurrentProcessImmediately(int exit_code)90*6777b538SAndroid Build Coastguard Worker void Process::TerminateCurrentProcessImmediately(int exit_code) {
91*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(CLANG_PROFILING)
92*6777b538SAndroid Build Coastguard Worker   WriteClangProfilingProfile();
93*6777b538SAndroid Build Coastguard Worker #endif
94*6777b538SAndroid Build Coastguard Worker   ::TerminateProcess(GetCurrentProcess(), static_cast<UINT>(exit_code));
95*6777b538SAndroid Build Coastguard Worker   // There is some ambiguity over whether the call above can return. Rather than
96*6777b538SAndroid Build Coastguard Worker   // hitting confusing crashes later on we should crash right here.
97*6777b538SAndroid Build Coastguard Worker   ImmediateCrash();
98*6777b538SAndroid Build Coastguard Worker }
99*6777b538SAndroid Build Coastguard Worker 
IsValid() const100*6777b538SAndroid Build Coastguard Worker bool Process::IsValid() const {
101*6777b538SAndroid Build Coastguard Worker   return process_.is_valid() || is_current();
102*6777b538SAndroid Build Coastguard Worker }
103*6777b538SAndroid Build Coastguard Worker 
Handle() const104*6777b538SAndroid Build Coastguard Worker ProcessHandle Process::Handle() const {
105*6777b538SAndroid Build Coastguard Worker   return is_current_process_ ? GetCurrentProcess() : process_.get();
106*6777b538SAndroid Build Coastguard Worker }
107*6777b538SAndroid Build Coastguard Worker 
Duplicate() const108*6777b538SAndroid Build Coastguard Worker Process Process::Duplicate() const {
109*6777b538SAndroid Build Coastguard Worker   if (is_current())
110*6777b538SAndroid Build Coastguard Worker     return Current();
111*6777b538SAndroid Build Coastguard Worker 
112*6777b538SAndroid Build Coastguard Worker   ProcessHandle out_handle;
113*6777b538SAndroid Build Coastguard Worker   if (!IsValid() || !::DuplicateHandle(GetCurrentProcess(),
114*6777b538SAndroid Build Coastguard Worker                                        Handle(),
115*6777b538SAndroid Build Coastguard Worker                                        GetCurrentProcess(),
116*6777b538SAndroid Build Coastguard Worker                                        &out_handle,
117*6777b538SAndroid Build Coastguard Worker                                        0,
118*6777b538SAndroid Build Coastguard Worker                                        FALSE,
119*6777b538SAndroid Build Coastguard Worker                                        DUPLICATE_SAME_ACCESS)) {
120*6777b538SAndroid Build Coastguard Worker     return Process();
121*6777b538SAndroid Build Coastguard Worker   }
122*6777b538SAndroid Build Coastguard Worker   return Process(out_handle);
123*6777b538SAndroid Build Coastguard Worker }
124*6777b538SAndroid Build Coastguard Worker 
Release()125*6777b538SAndroid Build Coastguard Worker ProcessHandle Process::Release() {
126*6777b538SAndroid Build Coastguard Worker   if (is_current())
127*6777b538SAndroid Build Coastguard Worker     return ::GetCurrentProcess();
128*6777b538SAndroid Build Coastguard Worker   return process_.release();
129*6777b538SAndroid Build Coastguard Worker }
130*6777b538SAndroid Build Coastguard Worker 
Pid() const131*6777b538SAndroid Build Coastguard Worker ProcessId Process::Pid() const {
132*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
133*6777b538SAndroid Build Coastguard Worker   return GetProcId(Handle());
134*6777b538SAndroid Build Coastguard Worker }
135*6777b538SAndroid Build Coastguard Worker 
CreationTime() const136*6777b538SAndroid Build Coastguard Worker Time Process::CreationTime() const {
137*6777b538SAndroid Build Coastguard Worker   FILETIME creation_time = {};
138*6777b538SAndroid Build Coastguard Worker   FILETIME ignore1 = {};
139*6777b538SAndroid Build Coastguard Worker   FILETIME ignore2 = {};
140*6777b538SAndroid Build Coastguard Worker   FILETIME ignore3 = {};
141*6777b538SAndroid Build Coastguard Worker   if (!::GetProcessTimes(Handle(), &creation_time, &ignore1, &ignore2,
142*6777b538SAndroid Build Coastguard Worker                          &ignore3)) {
143*6777b538SAndroid Build Coastguard Worker     return Time();
144*6777b538SAndroid Build Coastguard Worker   }
145*6777b538SAndroid Build Coastguard Worker   return Time::FromFileTime(creation_time);
146*6777b538SAndroid Build Coastguard Worker }
147*6777b538SAndroid Build Coastguard Worker 
is_current() const148*6777b538SAndroid Build Coastguard Worker bool Process::is_current() const {
149*6777b538SAndroid Build Coastguard Worker   return is_current_process_;
150*6777b538SAndroid Build Coastguard Worker }
151*6777b538SAndroid Build Coastguard Worker 
Close()152*6777b538SAndroid Build Coastguard Worker void Process::Close() {
153*6777b538SAndroid Build Coastguard Worker   is_current_process_ = false;
154*6777b538SAndroid Build Coastguard Worker   if (!process_.is_valid())
155*6777b538SAndroid Build Coastguard Worker     return;
156*6777b538SAndroid Build Coastguard Worker 
157*6777b538SAndroid Build Coastguard Worker   process_.Close();
158*6777b538SAndroid Build Coastguard Worker }
159*6777b538SAndroid Build Coastguard Worker 
Terminate(int exit_code,bool wait) const160*6777b538SAndroid Build Coastguard Worker bool Process::Terminate(int exit_code, bool wait) const {
161*6777b538SAndroid Build Coastguard Worker   constexpr DWORD kWaitMs = 60 * 1000;
162*6777b538SAndroid Build Coastguard Worker 
163*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
164*6777b538SAndroid Build Coastguard Worker   bool result =
165*6777b538SAndroid Build Coastguard Worker       ::TerminateProcess(Handle(), static_cast<UINT>(exit_code)) != FALSE;
166*6777b538SAndroid Build Coastguard Worker   if (result) {
167*6777b538SAndroid Build Coastguard Worker     // The process may not end immediately due to pending I/O
168*6777b538SAndroid Build Coastguard Worker     if (wait && ::WaitForSingleObject(Handle(), kWaitMs) != WAIT_OBJECT_0)
169*6777b538SAndroid Build Coastguard Worker       DPLOG(ERROR) << "Error waiting for process exit";
170*6777b538SAndroid Build Coastguard Worker     Exited(exit_code);
171*6777b538SAndroid Build Coastguard Worker   } else {
172*6777b538SAndroid Build Coastguard Worker     // The process can't be terminated, perhaps because it has already exited or
173*6777b538SAndroid Build Coastguard Worker     // is in the process of exiting. An error code of ERROR_ACCESS_DENIED is the
174*6777b538SAndroid Build Coastguard Worker     // undocumented-but-expected result if the process has already exited or
175*6777b538SAndroid Build Coastguard Worker     // started exiting when TerminateProcess is called, so don't print an error
176*6777b538SAndroid Build Coastguard Worker     // message in that case.
177*6777b538SAndroid Build Coastguard Worker     if (GetLastError() != ERROR_ACCESS_DENIED)
178*6777b538SAndroid Build Coastguard Worker       DPLOG(ERROR) << "Unable to terminate process";
179*6777b538SAndroid Build Coastguard Worker     // A non-zero timeout is necessary here for the same reasons as above.
180*6777b538SAndroid Build Coastguard Worker     if (::WaitForSingleObject(Handle(), kWaitMs) == WAIT_OBJECT_0) {
181*6777b538SAndroid Build Coastguard Worker       DWORD actual_exit;
182*6777b538SAndroid Build Coastguard Worker       Exited(::GetExitCodeProcess(Handle(), &actual_exit)
183*6777b538SAndroid Build Coastguard Worker                  ? static_cast<int>(actual_exit)
184*6777b538SAndroid Build Coastguard Worker                  : exit_code);
185*6777b538SAndroid Build Coastguard Worker       result = true;
186*6777b538SAndroid Build Coastguard Worker     }
187*6777b538SAndroid Build Coastguard Worker   }
188*6777b538SAndroid Build Coastguard Worker   return result;
189*6777b538SAndroid Build Coastguard Worker }
190*6777b538SAndroid Build Coastguard Worker 
WaitForExitOrEvent(const base::win::ScopedHandle & stop_event_handle,int * exit_code) const191*6777b538SAndroid Build Coastguard Worker Process::WaitExitStatus Process::WaitForExitOrEvent(
192*6777b538SAndroid Build Coastguard Worker     const base::win::ScopedHandle& stop_event_handle,
193*6777b538SAndroid Build Coastguard Worker     int* exit_code) const {
194*6777b538SAndroid Build Coastguard Worker   HANDLE events[] = {Handle(), stop_event_handle.get()};
195*6777b538SAndroid Build Coastguard Worker   DWORD wait_result =
196*6777b538SAndroid Build Coastguard Worker       ::WaitForMultipleObjects(std::size(events), events, FALSE, INFINITE);
197*6777b538SAndroid Build Coastguard Worker 
198*6777b538SAndroid Build Coastguard Worker   if (wait_result == WAIT_OBJECT_0) {
199*6777b538SAndroid Build Coastguard Worker     DWORD temp_code;  // Don't clobber out-parameters in case of failure.
200*6777b538SAndroid Build Coastguard Worker     if (!::GetExitCodeProcess(Handle(), &temp_code))
201*6777b538SAndroid Build Coastguard Worker       return Process::WaitExitStatus::FAILED;
202*6777b538SAndroid Build Coastguard Worker 
203*6777b538SAndroid Build Coastguard Worker     if (exit_code)
204*6777b538SAndroid Build Coastguard Worker       *exit_code = static_cast<int>(temp_code);
205*6777b538SAndroid Build Coastguard Worker 
206*6777b538SAndroid Build Coastguard Worker     Exited(static_cast<int>(temp_code));
207*6777b538SAndroid Build Coastguard Worker     return Process::WaitExitStatus::PROCESS_EXITED;
208*6777b538SAndroid Build Coastguard Worker   }
209*6777b538SAndroid Build Coastguard Worker 
210*6777b538SAndroid Build Coastguard Worker   if (wait_result == WAIT_OBJECT_0 + 1) {
211*6777b538SAndroid Build Coastguard Worker     return Process::WaitExitStatus::STOP_EVENT_SIGNALED;
212*6777b538SAndroid Build Coastguard Worker   }
213*6777b538SAndroid Build Coastguard Worker 
214*6777b538SAndroid Build Coastguard Worker   return Process::WaitExitStatus::FAILED;
215*6777b538SAndroid Build Coastguard Worker }
216*6777b538SAndroid Build Coastguard Worker 
WaitForExit(int * exit_code) const217*6777b538SAndroid Build Coastguard Worker bool Process::WaitForExit(int* exit_code) const {
218*6777b538SAndroid Build Coastguard Worker   return WaitForExitWithTimeout(TimeDelta::Max(), exit_code);
219*6777b538SAndroid Build Coastguard Worker }
220*6777b538SAndroid Build Coastguard Worker 
WaitForExitWithTimeout(TimeDelta timeout,int * exit_code) const221*6777b538SAndroid Build Coastguard Worker bool Process::WaitForExitWithTimeout(TimeDelta timeout, int* exit_code) const {
222*6777b538SAndroid Build Coastguard Worker   TRACE_EVENT0("base", "Process::WaitForExitWithTimeout");
223*6777b538SAndroid Build Coastguard Worker 
224*6777b538SAndroid Build Coastguard Worker   if (!timeout.is_zero()) {
225*6777b538SAndroid Build Coastguard Worker     // Assert that this thread is allowed to wait below. This intentionally
226*6777b538SAndroid Build Coastguard Worker     // doesn't use ScopedBlockingCallWithBaseSyncPrimitives because the process
227*6777b538SAndroid Build Coastguard Worker     // being waited upon tends to itself be using the CPU and considering this
228*6777b538SAndroid Build Coastguard Worker     // thread non-busy causes more issue than it fixes: http://crbug.com/905788
229*6777b538SAndroid Build Coastguard Worker     internal::AssertBaseSyncPrimitivesAllowed();
230*6777b538SAndroid Build Coastguard Worker   }
231*6777b538SAndroid Build Coastguard Worker 
232*6777b538SAndroid Build Coastguard Worker   // Limit timeout to INFINITE.
233*6777b538SAndroid Build Coastguard Worker   DWORD timeout_ms = saturated_cast<DWORD>(timeout.InMilliseconds());
234*6777b538SAndroid Build Coastguard Worker   if (::WaitForSingleObject(Handle(), timeout_ms) != WAIT_OBJECT_0)
235*6777b538SAndroid Build Coastguard Worker     return false;
236*6777b538SAndroid Build Coastguard Worker 
237*6777b538SAndroid Build Coastguard Worker   DWORD temp_code;  // Don't clobber out-parameters in case of failure.
238*6777b538SAndroid Build Coastguard Worker   if (!::GetExitCodeProcess(Handle(), &temp_code))
239*6777b538SAndroid Build Coastguard Worker     return false;
240*6777b538SAndroid Build Coastguard Worker 
241*6777b538SAndroid Build Coastguard Worker   if (exit_code)
242*6777b538SAndroid Build Coastguard Worker     *exit_code = static_cast<int>(temp_code);
243*6777b538SAndroid Build Coastguard Worker 
244*6777b538SAndroid Build Coastguard Worker   Exited(static_cast<int>(temp_code));
245*6777b538SAndroid Build Coastguard Worker   return true;
246*6777b538SAndroid Build Coastguard Worker }
247*6777b538SAndroid Build Coastguard Worker 
Exited(int exit_code) const248*6777b538SAndroid Build Coastguard Worker void Process::Exited(int exit_code) const {}
249*6777b538SAndroid Build Coastguard Worker 
GetPriority() const250*6777b538SAndroid Build Coastguard Worker Process::Priority Process::GetPriority() const {
251*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
252*6777b538SAndroid Build Coastguard Worker   int priority = GetOSPriority();
253*6777b538SAndroid Build Coastguard Worker   if (priority == 0)
254*6777b538SAndroid Build Coastguard Worker     return Priority::kUserBlocking;  // Failure case. Use default value.
255*6777b538SAndroid Build Coastguard Worker   if ((priority == BELOW_NORMAL_PRIORITY_CLASS) ||
256*6777b538SAndroid Build Coastguard Worker       (priority == IDLE_PRIORITY_CLASS)) {
257*6777b538SAndroid Build Coastguard Worker     return Priority::kBestEffort;
258*6777b538SAndroid Build Coastguard Worker   }
259*6777b538SAndroid Build Coastguard Worker   return Priority::kUserBlocking;
260*6777b538SAndroid Build Coastguard Worker }
261*6777b538SAndroid Build Coastguard Worker 
SetPriority(Priority priority)262*6777b538SAndroid Build Coastguard Worker bool Process::SetPriority(Priority priority) {
263*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
264*6777b538SAndroid Build Coastguard Worker   // Having a process remove itself from background mode is a potential
265*6777b538SAndroid Build Coastguard Worker   // priority inversion, and having a process put itself in background mode is
266*6777b538SAndroid Build Coastguard Worker   // broken in Windows 11 22H2. So, it is no longer supported. See
267*6777b538SAndroid Build Coastguard Worker   // https://crbug.com/1396155 for details.
268*6777b538SAndroid Build Coastguard Worker   DCHECK(!is_current());
269*6777b538SAndroid Build Coastguard Worker   const DWORD priority_class = priority == Priority::kBestEffort
270*6777b538SAndroid Build Coastguard Worker                                    ? IDLE_PRIORITY_CLASS
271*6777b538SAndroid Build Coastguard Worker                                    : NORMAL_PRIORITY_CLASS;
272*6777b538SAndroid Build Coastguard Worker 
273*6777b538SAndroid Build Coastguard Worker   if (base::win::OSInfo::GetInstance()->version() >=
274*6777b538SAndroid Build Coastguard Worker           base::win::Version::WIN11 &&
275*6777b538SAndroid Build Coastguard Worker       FeatureList::IsEnabled(kUseEcoQoSForBackgroundProcess)) {
276*6777b538SAndroid Build Coastguard Worker     PROCESS_POWER_THROTTLING_STATE power_throttling;
277*6777b538SAndroid Build Coastguard Worker     RtlZeroMemory(&power_throttling, sizeof(power_throttling));
278*6777b538SAndroid Build Coastguard Worker     power_throttling.Version = PROCESS_POWER_THROTTLING_CURRENT_VERSION;
279*6777b538SAndroid Build Coastguard Worker 
280*6777b538SAndroid Build Coastguard Worker     if (priority == Priority::kBestEffort) {
281*6777b538SAndroid Build Coastguard Worker       // Sets Eco QoS level.
282*6777b538SAndroid Build Coastguard Worker       power_throttling.ControlMask = PROCESS_POWER_THROTTLING_EXECUTION_SPEED;
283*6777b538SAndroid Build Coastguard Worker       power_throttling.StateMask = PROCESS_POWER_THROTTLING_EXECUTION_SPEED;
284*6777b538SAndroid Build Coastguard Worker     } else {
285*6777b538SAndroid Build Coastguard Worker       // Uses system default.
286*6777b538SAndroid Build Coastguard Worker       power_throttling.ControlMask = 0;
287*6777b538SAndroid Build Coastguard Worker       power_throttling.StateMask = 0;
288*6777b538SAndroid Build Coastguard Worker     }
289*6777b538SAndroid Build Coastguard Worker     bool ret =
290*6777b538SAndroid Build Coastguard Worker         ::SetProcessInformation(Handle(), ProcessPowerThrottling,
291*6777b538SAndroid Build Coastguard Worker                                 &power_throttling, sizeof(power_throttling));
292*6777b538SAndroid Build Coastguard Worker     if (ret == 0) {
293*6777b538SAndroid Build Coastguard Worker       DPLOG(ERROR) << "Setting process QoS policy fails";
294*6777b538SAndroid Build Coastguard Worker     }
295*6777b538SAndroid Build Coastguard Worker   }
296*6777b538SAndroid Build Coastguard Worker 
297*6777b538SAndroid Build Coastguard Worker   return (::SetPriorityClass(Handle(), priority_class) != 0);
298*6777b538SAndroid Build Coastguard Worker }
299*6777b538SAndroid Build Coastguard Worker 
GetOSPriority() const300*6777b538SAndroid Build Coastguard Worker int Process::GetOSPriority() const {
301*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
302*6777b538SAndroid Build Coastguard Worker   return static_cast<int>(::GetPriorityClass(Handle()));
303*6777b538SAndroid Build Coastguard Worker }
304*6777b538SAndroid Build Coastguard Worker 
305*6777b538SAndroid Build Coastguard Worker }  // namespace base
306