xref: /aosp_15_r20/external/cronet/base/process/process_fuchsia.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2017 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 <lib/zx/process.h>
8*6777b538SAndroid Build Coastguard Worker #include <zircon/process.h>
9*6777b538SAndroid Build Coastguard Worker #include <zircon/syscalls.h>
10*6777b538SAndroid Build Coastguard Worker 
11*6777b538SAndroid Build Coastguard Worker #include "base/clang_profiling_buildflags.h"
12*6777b538SAndroid Build Coastguard Worker #include "base/fuchsia/default_job.h"
13*6777b538SAndroid Build Coastguard Worker #include "base/fuchsia/fuchsia_logging.h"
14*6777b538SAndroid Build Coastguard Worker #include "base/threading/thread_restrictions.h"
15*6777b538SAndroid Build Coastguard Worker #include "base/trace_event/base_tracing.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 base {
22*6777b538SAndroid Build Coastguard Worker 
23*6777b538SAndroid Build Coastguard Worker namespace {
24*6777b538SAndroid Build Coastguard Worker 
FindProcessInJobTree(const zx::job & job,ProcessId pid)25*6777b538SAndroid Build Coastguard Worker zx::process FindProcessInJobTree(const zx::job& job, ProcessId pid) {
26*6777b538SAndroid Build Coastguard Worker   zx::process process;
27*6777b538SAndroid Build Coastguard Worker   zx_status_t status = job.get_child(pid, ZX_RIGHT_SAME_RIGHTS, &process);
28*6777b538SAndroid Build Coastguard Worker 
29*6777b538SAndroid Build Coastguard Worker   if (status == ZX_OK)
30*6777b538SAndroid Build Coastguard Worker     return process;
31*6777b538SAndroid Build Coastguard Worker 
32*6777b538SAndroid Build Coastguard Worker   if (status == ZX_ERR_NOT_FOUND) {
33*6777b538SAndroid Build Coastguard Worker     std::vector<zx_koid_t> job_koids(32);
34*6777b538SAndroid Build Coastguard Worker     while (true) {
35*6777b538SAndroid Build Coastguard Worker       // Fetch the KOIDs of the job children of |job|.
36*6777b538SAndroid Build Coastguard Worker       size_t actual = 0u;
37*6777b538SAndroid Build Coastguard Worker       size_t available = 0u;
38*6777b538SAndroid Build Coastguard Worker       status = job.get_info(ZX_INFO_JOB_CHILDREN, job_koids.data(),
39*6777b538SAndroid Build Coastguard Worker                             job_koids.size() * sizeof(zx_koid_t), &actual,
40*6777b538SAndroid Build Coastguard Worker                             &available);
41*6777b538SAndroid Build Coastguard Worker 
42*6777b538SAndroid Build Coastguard Worker       if (status != ZX_OK) {
43*6777b538SAndroid Build Coastguard Worker         ZX_DLOG(ERROR, status) << "zx_object_get_info(JOB_CHILDREN)";
44*6777b538SAndroid Build Coastguard Worker         return zx::process();
45*6777b538SAndroid Build Coastguard Worker       }
46*6777b538SAndroid Build Coastguard Worker 
47*6777b538SAndroid Build Coastguard Worker       // If |job_koids| was too small then resize it and try again.
48*6777b538SAndroid Build Coastguard Worker       if (available > actual) {
49*6777b538SAndroid Build Coastguard Worker         job_koids.resize(available);
50*6777b538SAndroid Build Coastguard Worker         continue;
51*6777b538SAndroid Build Coastguard Worker       }
52*6777b538SAndroid Build Coastguard Worker 
53*6777b538SAndroid Build Coastguard Worker       // Break out of the loop and iterate over |job_koids|, to find the PID.
54*6777b538SAndroid Build Coastguard Worker       job_koids.resize(actual);
55*6777b538SAndroid Build Coastguard Worker       break;
56*6777b538SAndroid Build Coastguard Worker     }
57*6777b538SAndroid Build Coastguard Worker 
58*6777b538SAndroid Build Coastguard Worker     for (zx_koid_t job_koid : job_koids) {
59*6777b538SAndroid Build Coastguard Worker       zx::job child_job;
60*6777b538SAndroid Build Coastguard Worker       if (job.get_child(job_koid, ZX_RIGHT_SAME_RIGHTS, &child_job) != ZX_OK)
61*6777b538SAndroid Build Coastguard Worker         continue;
62*6777b538SAndroid Build Coastguard Worker       process = FindProcessInJobTree(child_job, pid);
63*6777b538SAndroid Build Coastguard Worker       if (process)
64*6777b538SAndroid Build Coastguard Worker         return process;
65*6777b538SAndroid Build Coastguard Worker     }
66*6777b538SAndroid Build Coastguard Worker 
67*6777b538SAndroid Build Coastguard Worker     return zx::process();
68*6777b538SAndroid Build Coastguard Worker   }
69*6777b538SAndroid Build Coastguard Worker 
70*6777b538SAndroid Build Coastguard Worker   ZX_DLOG(ERROR, status) << "zx_object_get_child";
71*6777b538SAndroid Build Coastguard Worker   return zx::process();
72*6777b538SAndroid Build Coastguard Worker }
73*6777b538SAndroid Build Coastguard Worker 
74*6777b538SAndroid Build Coastguard Worker }  // namespace
75*6777b538SAndroid Build Coastguard Worker 
Process(ProcessHandle handle)76*6777b538SAndroid Build Coastguard Worker Process::Process(ProcessHandle handle)
77*6777b538SAndroid Build Coastguard Worker     : process_(handle), is_current_process_(false) {
78*6777b538SAndroid Build Coastguard Worker   CHECK_NE(handle, zx_process_self());
79*6777b538SAndroid Build Coastguard Worker }
80*6777b538SAndroid Build Coastguard Worker 
~Process()81*6777b538SAndroid Build Coastguard Worker Process::~Process() {
82*6777b538SAndroid Build Coastguard Worker   Close();
83*6777b538SAndroid Build Coastguard Worker }
84*6777b538SAndroid Build Coastguard Worker 
Process(Process && other)85*6777b538SAndroid Build Coastguard Worker Process::Process(Process&& other)
86*6777b538SAndroid Build Coastguard Worker     : process_(std::move(other.process_)),
87*6777b538SAndroid Build Coastguard Worker       is_current_process_(other.is_current_process_) {
88*6777b538SAndroid Build Coastguard Worker   other.is_current_process_ = false;
89*6777b538SAndroid Build Coastguard Worker }
90*6777b538SAndroid Build Coastguard Worker 
operator =(Process && other)91*6777b538SAndroid Build Coastguard Worker Process& Process::operator=(Process&& other) {
92*6777b538SAndroid Build Coastguard Worker   process_ = std::move(other.process_);
93*6777b538SAndroid Build Coastguard Worker   is_current_process_ = other.is_current_process_;
94*6777b538SAndroid Build Coastguard Worker   other.is_current_process_ = false;
95*6777b538SAndroid Build Coastguard Worker   return *this;
96*6777b538SAndroid Build Coastguard Worker }
97*6777b538SAndroid Build Coastguard Worker 
98*6777b538SAndroid Build Coastguard Worker // static
Current()99*6777b538SAndroid Build Coastguard Worker Process Process::Current() {
100*6777b538SAndroid Build Coastguard Worker   Process process;
101*6777b538SAndroid Build Coastguard Worker   process.is_current_process_ = true;
102*6777b538SAndroid Build Coastguard Worker   return process;
103*6777b538SAndroid Build Coastguard Worker }
104*6777b538SAndroid Build Coastguard Worker 
105*6777b538SAndroid Build Coastguard Worker // static
Open(ProcessId pid)106*6777b538SAndroid Build Coastguard Worker Process Process::Open(ProcessId pid) {
107*6777b538SAndroid Build Coastguard Worker   if (pid == GetCurrentProcId())
108*6777b538SAndroid Build Coastguard Worker     return Current();
109*6777b538SAndroid Build Coastguard Worker 
110*6777b538SAndroid Build Coastguard Worker   return Process(FindProcessInJobTree(*GetDefaultJob(), pid).release());
111*6777b538SAndroid Build Coastguard Worker }
112*6777b538SAndroid Build Coastguard Worker 
113*6777b538SAndroid Build Coastguard Worker // static
OpenWithExtraPrivileges(ProcessId pid)114*6777b538SAndroid Build Coastguard Worker Process Process::OpenWithExtraPrivileges(ProcessId pid) {
115*6777b538SAndroid Build Coastguard Worker   // No privileges to set.
116*6777b538SAndroid Build Coastguard Worker   return Open(pid);
117*6777b538SAndroid Build Coastguard Worker }
118*6777b538SAndroid Build Coastguard Worker 
119*6777b538SAndroid Build Coastguard Worker // static
CanSetPriority()120*6777b538SAndroid Build Coastguard Worker bool Process::CanSetPriority() {
121*6777b538SAndroid Build Coastguard Worker   return false;
122*6777b538SAndroid Build Coastguard Worker }
123*6777b538SAndroid Build Coastguard Worker 
124*6777b538SAndroid Build Coastguard Worker // static
TerminateCurrentProcessImmediately(int exit_code)125*6777b538SAndroid Build Coastguard Worker void Process::TerminateCurrentProcessImmediately(int exit_code) {
126*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(CLANG_PROFILING)
127*6777b538SAndroid Build Coastguard Worker   WriteClangProfilingProfile();
128*6777b538SAndroid Build Coastguard Worker #endif
129*6777b538SAndroid Build Coastguard Worker   _exit(exit_code);
130*6777b538SAndroid Build Coastguard Worker }
131*6777b538SAndroid Build Coastguard Worker 
IsValid() const132*6777b538SAndroid Build Coastguard Worker bool Process::IsValid() const {
133*6777b538SAndroid Build Coastguard Worker   return process_.is_valid() || is_current();
134*6777b538SAndroid Build Coastguard Worker }
135*6777b538SAndroid Build Coastguard Worker 
Handle() const136*6777b538SAndroid Build Coastguard Worker ProcessHandle Process::Handle() const {
137*6777b538SAndroid Build Coastguard Worker   return is_current_process_ ? zx_process_self() : process_.get();
138*6777b538SAndroid Build Coastguard Worker }
139*6777b538SAndroid Build Coastguard Worker 
Duplicate() const140*6777b538SAndroid Build Coastguard Worker Process Process::Duplicate() const {
141*6777b538SAndroid Build Coastguard Worker   if (is_current())
142*6777b538SAndroid Build Coastguard Worker     return Current();
143*6777b538SAndroid Build Coastguard Worker 
144*6777b538SAndroid Build Coastguard Worker   if (!IsValid())
145*6777b538SAndroid Build Coastguard Worker     return Process();
146*6777b538SAndroid Build Coastguard Worker 
147*6777b538SAndroid Build Coastguard Worker   zx::process out;
148*6777b538SAndroid Build Coastguard Worker   zx_status_t result = process_.duplicate(ZX_RIGHT_SAME_RIGHTS, &out);
149*6777b538SAndroid Build Coastguard Worker   if (result != ZX_OK) {
150*6777b538SAndroid Build Coastguard Worker     ZX_DLOG(ERROR, result) << "zx_handle_duplicate";
151*6777b538SAndroid Build Coastguard Worker     return Process();
152*6777b538SAndroid Build Coastguard Worker   }
153*6777b538SAndroid Build Coastguard Worker 
154*6777b538SAndroid Build Coastguard Worker   return Process(out.release());
155*6777b538SAndroid Build Coastguard Worker }
156*6777b538SAndroid Build Coastguard Worker 
Release()157*6777b538SAndroid Build Coastguard Worker ProcessHandle Process::Release() {
158*6777b538SAndroid Build Coastguard Worker   if (is_current()) {
159*6777b538SAndroid Build Coastguard Worker     // Caller expects to own the reference, so duplicate the self handle.
160*6777b538SAndroid Build Coastguard Worker     zx::process handle;
161*6777b538SAndroid Build Coastguard Worker     zx_status_t result =
162*6777b538SAndroid Build Coastguard Worker         zx::process::self()->duplicate(ZX_RIGHT_SAME_RIGHTS, &handle);
163*6777b538SAndroid Build Coastguard Worker     if (result != ZX_OK) {
164*6777b538SAndroid Build Coastguard Worker       return kNullProcessHandle;
165*6777b538SAndroid Build Coastguard Worker     }
166*6777b538SAndroid Build Coastguard Worker     is_current_process_ = false;
167*6777b538SAndroid Build Coastguard Worker     return handle.release();
168*6777b538SAndroid Build Coastguard Worker   }
169*6777b538SAndroid Build Coastguard Worker   return process_.release();
170*6777b538SAndroid Build Coastguard Worker }
171*6777b538SAndroid Build Coastguard Worker 
Pid() const172*6777b538SAndroid Build Coastguard Worker ProcessId Process::Pid() const {
173*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
174*6777b538SAndroid Build Coastguard Worker   return GetProcId(Handle());
175*6777b538SAndroid Build Coastguard Worker }
176*6777b538SAndroid Build Coastguard Worker 
CreationTime() const177*6777b538SAndroid Build Coastguard Worker Time Process::CreationTime() const {
178*6777b538SAndroid Build Coastguard Worker   zx_info_process_t proc_info;
179*6777b538SAndroid Build Coastguard Worker   zx_status_t status =
180*6777b538SAndroid Build Coastguard Worker       zx_object_get_info(Handle(), ZX_INFO_PROCESS, &proc_info,
181*6777b538SAndroid Build Coastguard Worker                          sizeof(proc_info), nullptr, nullptr);
182*6777b538SAndroid Build Coastguard Worker   if (status != ZX_OK) {
183*6777b538SAndroid Build Coastguard Worker     ZX_DLOG(ERROR, status) << "zx_process_get_info";
184*6777b538SAndroid Build Coastguard Worker     return Time();
185*6777b538SAndroid Build Coastguard Worker   }
186*6777b538SAndroid Build Coastguard Worker   if ((proc_info.flags & ZX_INFO_PROCESS_FLAG_STARTED) == 0) {
187*6777b538SAndroid Build Coastguard Worker     DLOG(WARNING) << "zx_process_get_info: Not started.";
188*6777b538SAndroid Build Coastguard Worker     return Time();
189*6777b538SAndroid Build Coastguard Worker   }
190*6777b538SAndroid Build Coastguard Worker   // Process creation times are expressed in ticks since system boot, so
191*6777b538SAndroid Build Coastguard Worker   // perform a best-effort translation from that to UTC "wall-clock" time.
192*6777b538SAndroid Build Coastguard Worker   return Time::Now() +
193*6777b538SAndroid Build Coastguard Worker          (TimeTicks::FromZxTime(proc_info.start_time) - TimeTicks::Now());
194*6777b538SAndroid Build Coastguard Worker }
195*6777b538SAndroid Build Coastguard Worker 
is_current() const196*6777b538SAndroid Build Coastguard Worker bool Process::is_current() const {
197*6777b538SAndroid Build Coastguard Worker   return is_current_process_;
198*6777b538SAndroid Build Coastguard Worker }
199*6777b538SAndroid Build Coastguard Worker 
Close()200*6777b538SAndroid Build Coastguard Worker void Process::Close() {
201*6777b538SAndroid Build Coastguard Worker   is_current_process_ = false;
202*6777b538SAndroid Build Coastguard Worker   process_.reset();
203*6777b538SAndroid Build Coastguard Worker }
204*6777b538SAndroid Build Coastguard Worker 
Terminate(int exit_code,bool wait) const205*6777b538SAndroid Build Coastguard Worker bool Process::Terminate(int exit_code, bool wait) const {
206*6777b538SAndroid Build Coastguard Worker   // exit_code isn't supportable. https://crbug.com/753490.
207*6777b538SAndroid Build Coastguard Worker   zx_status_t status = zx_task_kill(Handle());
208*6777b538SAndroid Build Coastguard Worker   if (status == ZX_OK && wait) {
209*6777b538SAndroid Build Coastguard Worker     zx_signals_t signals;
210*6777b538SAndroid Build Coastguard Worker     status = zx_object_wait_one(Handle(), ZX_TASK_TERMINATED,
211*6777b538SAndroid Build Coastguard Worker                                 zx_deadline_after(ZX_SEC(60)), &signals);
212*6777b538SAndroid Build Coastguard Worker     if (status != ZX_OK) {
213*6777b538SAndroid Build Coastguard Worker       ZX_DLOG(ERROR, status) << "zx_object_wait_one(terminate)";
214*6777b538SAndroid Build Coastguard Worker     } else {
215*6777b538SAndroid Build Coastguard Worker       CHECK(signals & ZX_TASK_TERMINATED);
216*6777b538SAndroid Build Coastguard Worker     }
217*6777b538SAndroid Build Coastguard Worker   } else if (status != ZX_OK) {
218*6777b538SAndroid Build Coastguard Worker     ZX_DLOG(ERROR, status) << "zx_task_kill";
219*6777b538SAndroid Build Coastguard Worker   }
220*6777b538SAndroid Build Coastguard Worker 
221*6777b538SAndroid Build Coastguard Worker   return status >= 0;
222*6777b538SAndroid Build Coastguard Worker }
223*6777b538SAndroid Build Coastguard Worker 
WaitForExit(int * exit_code) const224*6777b538SAndroid Build Coastguard Worker bool Process::WaitForExit(int* exit_code) const {
225*6777b538SAndroid Build Coastguard Worker   return WaitForExitWithTimeout(TimeDelta::Max(), exit_code);
226*6777b538SAndroid Build Coastguard Worker }
227*6777b538SAndroid Build Coastguard Worker 
WaitForExitWithTimeout(TimeDelta timeout,int * exit_code) const228*6777b538SAndroid Build Coastguard Worker bool Process::WaitForExitWithTimeout(TimeDelta timeout, int* exit_code) const {
229*6777b538SAndroid Build Coastguard Worker   if (is_current_process_)
230*6777b538SAndroid Build Coastguard Worker     return false;
231*6777b538SAndroid Build Coastguard Worker 
232*6777b538SAndroid Build Coastguard Worker   TRACE_EVENT0("base", "Process::WaitForExitWithTimeout");
233*6777b538SAndroid Build Coastguard Worker 
234*6777b538SAndroid Build Coastguard Worker   if (!timeout.is_zero()) {
235*6777b538SAndroid Build Coastguard Worker     // Assert that this thread is allowed to wait below. This intentionally
236*6777b538SAndroid Build Coastguard Worker     // doesn't use ScopedBlockingCallWithBaseSyncPrimitives because the process
237*6777b538SAndroid Build Coastguard Worker     // being waited upon tends to itself be using the CPU and considering this
238*6777b538SAndroid Build Coastguard Worker     // thread non-busy causes more issue than it fixes: http://crbug.com/905788
239*6777b538SAndroid Build Coastguard Worker     internal::AssertBaseSyncPrimitivesAllowed();
240*6777b538SAndroid Build Coastguard Worker   }
241*6777b538SAndroid Build Coastguard Worker 
242*6777b538SAndroid Build Coastguard Worker   zx::time deadline = timeout == TimeDelta::Max()
243*6777b538SAndroid Build Coastguard Worker                           ? zx::time::infinite()
244*6777b538SAndroid Build Coastguard Worker                           : zx::time((TimeTicks::Now() + timeout).ToZxTime());
245*6777b538SAndroid Build Coastguard Worker   zx_signals_t signals_observed = 0;
246*6777b538SAndroid Build Coastguard Worker   zx_status_t status =
247*6777b538SAndroid Build Coastguard Worker       process_.wait_one(ZX_TASK_TERMINATED, deadline, &signals_observed);
248*6777b538SAndroid Build Coastguard Worker   if (status != ZX_OK) {
249*6777b538SAndroid Build Coastguard Worker     ZX_DLOG_IF(ERROR, status != ZX_ERR_TIMED_OUT, status)
250*6777b538SAndroid Build Coastguard Worker         << "zx_object_wait_one";
251*6777b538SAndroid Build Coastguard Worker     return false;
252*6777b538SAndroid Build Coastguard Worker   }
253*6777b538SAndroid Build Coastguard Worker 
254*6777b538SAndroid Build Coastguard Worker   zx_info_process_t proc_info;
255*6777b538SAndroid Build Coastguard Worker   status = process_.get_info(ZX_INFO_PROCESS, &proc_info, sizeof(proc_info),
256*6777b538SAndroid Build Coastguard Worker                              nullptr, nullptr);
257*6777b538SAndroid Build Coastguard Worker   if (status != ZX_OK) {
258*6777b538SAndroid Build Coastguard Worker     ZX_DLOG(ERROR, status) << "zx_object_get_info";
259*6777b538SAndroid Build Coastguard Worker     if (exit_code)
260*6777b538SAndroid Build Coastguard Worker       *exit_code = -1;
261*6777b538SAndroid Build Coastguard Worker     return false;
262*6777b538SAndroid Build Coastguard Worker   }
263*6777b538SAndroid Build Coastguard Worker 
264*6777b538SAndroid Build Coastguard Worker   if (exit_code)
265*6777b538SAndroid Build Coastguard Worker     *exit_code = static_cast<int>(proc_info.return_code);
266*6777b538SAndroid Build Coastguard Worker 
267*6777b538SAndroid Build Coastguard Worker   return true;
268*6777b538SAndroid Build Coastguard Worker }
269*6777b538SAndroid Build Coastguard Worker 
Exited(int exit_code) const270*6777b538SAndroid Build Coastguard Worker void Process::Exited(int exit_code) const {}
271*6777b538SAndroid Build Coastguard Worker 
GetPriority() const272*6777b538SAndroid Build Coastguard Worker Process::Priority Process::GetPriority() const {
273*6777b538SAndroid Build Coastguard Worker   // See SetPriority().
274*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
275*6777b538SAndroid Build Coastguard Worker   return Priority::kUserBlocking;
276*6777b538SAndroid Build Coastguard Worker }
277*6777b538SAndroid Build Coastguard Worker 
SetPriority(Priority priority)278*6777b538SAndroid Build Coastguard Worker bool Process::SetPriority(Priority priority) {
279*6777b538SAndroid Build Coastguard Worker   // No process priorities on Fuchsia.
280*6777b538SAndroid Build Coastguard Worker   // TODO(fxbug.dev/30735): Update this later if priorities are implemented.
281*6777b538SAndroid Build Coastguard Worker   return false;
282*6777b538SAndroid Build Coastguard Worker }
283*6777b538SAndroid Build Coastguard Worker 
GetOSPriority() const284*6777b538SAndroid Build Coastguard Worker int Process::GetOSPriority() const {
285*6777b538SAndroid Build Coastguard Worker   DCHECK(IsValid());
286*6777b538SAndroid Build Coastguard Worker   // No process priorities on Fuchsia.
287*6777b538SAndroid Build Coastguard Worker   return 0;
288*6777b538SAndroid Build Coastguard Worker }
289*6777b538SAndroid Build Coastguard Worker 
290*6777b538SAndroid Build Coastguard Worker }  // namespace base
291