xref: /aosp_15_r20/frameworks/native/cmds/dumpstate/DumpstateUtil.cpp (revision 38e8c45f13ce32b0dcecb25141ffecaf386fa17f)
1*38e8c45fSAndroid Build Coastguard Worker /*
2*38e8c45fSAndroid Build Coastguard Worker  * Copyright (C) 2016 The Android Open Source Project
3*38e8c45fSAndroid Build Coastguard Worker  *
4*38e8c45fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*38e8c45fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*38e8c45fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*38e8c45fSAndroid Build Coastguard Worker  *
8*38e8c45fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*38e8c45fSAndroid Build Coastguard Worker  *
10*38e8c45fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*38e8c45fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*38e8c45fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*38e8c45fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*38e8c45fSAndroid Build Coastguard Worker  * limitations under the License.
15*38e8c45fSAndroid Build Coastguard Worker  */
16*38e8c45fSAndroid Build Coastguard Worker 
17*38e8c45fSAndroid Build Coastguard Worker #define LOG_TAG "dumpstate"
18*38e8c45fSAndroid Build Coastguard Worker 
19*38e8c45fSAndroid Build Coastguard Worker #include "DumpstateUtil.h"
20*38e8c45fSAndroid Build Coastguard Worker 
21*38e8c45fSAndroid Build Coastguard Worker #include <dirent.h>
22*38e8c45fSAndroid Build Coastguard Worker #include <fcntl.h>
23*38e8c45fSAndroid Build Coastguard Worker #include <sys/prctl.h>
24*38e8c45fSAndroid Build Coastguard Worker #include <sys/wait.h>
25*38e8c45fSAndroid Build Coastguard Worker #include <unistd.h>
26*38e8c45fSAndroid Build Coastguard Worker 
27*38e8c45fSAndroid Build Coastguard Worker #include <vector>
28*38e8c45fSAndroid Build Coastguard Worker 
29*38e8c45fSAndroid Build Coastguard Worker #include <android-base/file.h>
30*38e8c45fSAndroid Build Coastguard Worker #include <android-base/properties.h>
31*38e8c45fSAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
32*38e8c45fSAndroid Build Coastguard Worker #include <android-base/strings.h>
33*38e8c45fSAndroid Build Coastguard Worker #include <android-base/unique_fd.h>
34*38e8c45fSAndroid Build Coastguard Worker #include <log/log.h>
35*38e8c45fSAndroid Build Coastguard Worker 
36*38e8c45fSAndroid Build Coastguard Worker #include "DumpstateInternal.h"
37*38e8c45fSAndroid Build Coastguard Worker 
38*38e8c45fSAndroid Build Coastguard Worker namespace android {
39*38e8c45fSAndroid Build Coastguard Worker namespace os {
40*38e8c45fSAndroid Build Coastguard Worker namespace dumpstate {
41*38e8c45fSAndroid Build Coastguard Worker 
42*38e8c45fSAndroid Build Coastguard Worker namespace {
43*38e8c45fSAndroid Build Coastguard Worker 
44*38e8c45fSAndroid Build Coastguard Worker static constexpr const char* kSuPath = "/system/xbin/su";
45*38e8c45fSAndroid Build Coastguard Worker 
waitpid_with_timeout(pid_t pid,int timeout_ms,int * status)46*38e8c45fSAndroid Build Coastguard Worker static bool waitpid_with_timeout(pid_t pid, int timeout_ms, int* status) {
47*38e8c45fSAndroid Build Coastguard Worker     sigset_t child_mask, old_mask;
48*38e8c45fSAndroid Build Coastguard Worker     sigemptyset(&child_mask);
49*38e8c45fSAndroid Build Coastguard Worker     sigaddset(&child_mask, SIGCHLD);
50*38e8c45fSAndroid Build Coastguard Worker 
51*38e8c45fSAndroid Build Coastguard Worker     // block SIGCHLD before we check if a process has exited
52*38e8c45fSAndroid Build Coastguard Worker     if (sigprocmask(SIG_BLOCK, &child_mask, &old_mask) == -1) {
53*38e8c45fSAndroid Build Coastguard Worker         printf("*** sigprocmask failed: %s\n", strerror(errno));
54*38e8c45fSAndroid Build Coastguard Worker         return false;
55*38e8c45fSAndroid Build Coastguard Worker     }
56*38e8c45fSAndroid Build Coastguard Worker 
57*38e8c45fSAndroid Build Coastguard Worker     // if the child has exited already, handle and reset signals before leaving
58*38e8c45fSAndroid Build Coastguard Worker     pid_t child_pid = waitpid(pid, status, WNOHANG);
59*38e8c45fSAndroid Build Coastguard Worker     if (child_pid != pid) {
60*38e8c45fSAndroid Build Coastguard Worker         if (child_pid > 0) {
61*38e8c45fSAndroid Build Coastguard Worker             printf("*** Waiting for pid %d, got pid %d instead\n", pid, child_pid);
62*38e8c45fSAndroid Build Coastguard Worker             sigprocmask(SIG_SETMASK, &old_mask, nullptr);
63*38e8c45fSAndroid Build Coastguard Worker             return false;
64*38e8c45fSAndroid Build Coastguard Worker         }
65*38e8c45fSAndroid Build Coastguard Worker     } else {
66*38e8c45fSAndroid Build Coastguard Worker         sigprocmask(SIG_SETMASK, &old_mask, nullptr);
67*38e8c45fSAndroid Build Coastguard Worker         return true;
68*38e8c45fSAndroid Build Coastguard Worker     }
69*38e8c45fSAndroid Build Coastguard Worker 
70*38e8c45fSAndroid Build Coastguard Worker     // wait for a SIGCHLD
71*38e8c45fSAndroid Build Coastguard Worker     timespec ts;
72*38e8c45fSAndroid Build Coastguard Worker     ts.tv_sec = MSEC_TO_SEC(timeout_ms);
73*38e8c45fSAndroid Build Coastguard Worker     ts.tv_nsec = (timeout_ms % 1000) * 1000000;
74*38e8c45fSAndroid Build Coastguard Worker     int ret = TEMP_FAILURE_RETRY(sigtimedwait(&child_mask, nullptr, &ts));
75*38e8c45fSAndroid Build Coastguard Worker     int saved_errno = errno;
76*38e8c45fSAndroid Build Coastguard Worker 
77*38e8c45fSAndroid Build Coastguard Worker     // Set the signals back the way they were.
78*38e8c45fSAndroid Build Coastguard Worker     if (sigprocmask(SIG_SETMASK, &old_mask, nullptr) == -1) {
79*38e8c45fSAndroid Build Coastguard Worker         printf("*** sigprocmask failed: %s\n", strerror(errno));
80*38e8c45fSAndroid Build Coastguard Worker         if (ret == 0) {
81*38e8c45fSAndroid Build Coastguard Worker             return false;
82*38e8c45fSAndroid Build Coastguard Worker         }
83*38e8c45fSAndroid Build Coastguard Worker     }
84*38e8c45fSAndroid Build Coastguard Worker     if (ret == -1) {
85*38e8c45fSAndroid Build Coastguard Worker         errno = saved_errno;
86*38e8c45fSAndroid Build Coastguard Worker         if (errno == EAGAIN) {
87*38e8c45fSAndroid Build Coastguard Worker             errno = ETIMEDOUT;
88*38e8c45fSAndroid Build Coastguard Worker         } else {
89*38e8c45fSAndroid Build Coastguard Worker             printf("*** sigtimedwait failed: %s\n", strerror(errno));
90*38e8c45fSAndroid Build Coastguard Worker         }
91*38e8c45fSAndroid Build Coastguard Worker         return false;
92*38e8c45fSAndroid Build Coastguard Worker     }
93*38e8c45fSAndroid Build Coastguard Worker 
94*38e8c45fSAndroid Build Coastguard Worker     child_pid = waitpid(pid, status, WNOHANG);
95*38e8c45fSAndroid Build Coastguard Worker     if (child_pid != pid) {
96*38e8c45fSAndroid Build Coastguard Worker         if (child_pid != -1) {
97*38e8c45fSAndroid Build Coastguard Worker             printf("*** Waiting for pid %d, got pid %d instead\n", pid, child_pid);
98*38e8c45fSAndroid Build Coastguard Worker         } else {
99*38e8c45fSAndroid Build Coastguard Worker             printf("*** waitpid failed: %s\n", strerror(errno));
100*38e8c45fSAndroid Build Coastguard Worker         }
101*38e8c45fSAndroid Build Coastguard Worker         return false;
102*38e8c45fSAndroid Build Coastguard Worker     }
103*38e8c45fSAndroid Build Coastguard Worker     return true;
104*38e8c45fSAndroid Build Coastguard Worker }
105*38e8c45fSAndroid Build Coastguard Worker }  // unnamed namespace
106*38e8c45fSAndroid Build Coastguard Worker 
107*38e8c45fSAndroid Build Coastguard Worker CommandOptions CommandOptions::DEFAULT = CommandOptions::WithTimeout(10).Build();
108*38e8c45fSAndroid Build Coastguard Worker CommandOptions CommandOptions::AS_ROOT = CommandOptions::WithTimeout(10).AsRoot().Build();
109*38e8c45fSAndroid Build Coastguard Worker 
CommandOptionsBuilder(int64_t timeout_ms)110*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder::CommandOptionsBuilder(int64_t timeout_ms) : values(timeout_ms) {
111*38e8c45fSAndroid Build Coastguard Worker }
112*38e8c45fSAndroid Build Coastguard Worker 
Always()113*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder& CommandOptions::CommandOptionsBuilder::Always() {
114*38e8c45fSAndroid Build Coastguard Worker     values.always_ = true;
115*38e8c45fSAndroid Build Coastguard Worker     return *this;
116*38e8c45fSAndroid Build Coastguard Worker }
117*38e8c45fSAndroid Build Coastguard Worker 
AsRoot()118*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder& CommandOptions::CommandOptionsBuilder::AsRoot() {
119*38e8c45fSAndroid Build Coastguard Worker     if (!PropertiesHelper::IsUnroot()) {
120*38e8c45fSAndroid Build Coastguard Worker         values.account_mode_ = SU_ROOT;
121*38e8c45fSAndroid Build Coastguard Worker     }
122*38e8c45fSAndroid Build Coastguard Worker     return *this;
123*38e8c45fSAndroid Build Coastguard Worker }
124*38e8c45fSAndroid Build Coastguard Worker 
AsRootIfAvailable()125*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder& CommandOptions::CommandOptionsBuilder::AsRootIfAvailable() {
126*38e8c45fSAndroid Build Coastguard Worker     if (!PropertiesHelper::IsUserBuild()) {
127*38e8c45fSAndroid Build Coastguard Worker         return AsRoot();
128*38e8c45fSAndroid Build Coastguard Worker     }
129*38e8c45fSAndroid Build Coastguard Worker     return *this;
130*38e8c45fSAndroid Build Coastguard Worker }
131*38e8c45fSAndroid Build Coastguard Worker 
DropRoot()132*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder& CommandOptions::CommandOptionsBuilder::DropRoot() {
133*38e8c45fSAndroid Build Coastguard Worker     values.account_mode_ = DROP_ROOT;
134*38e8c45fSAndroid Build Coastguard Worker     return *this;
135*38e8c45fSAndroid Build Coastguard Worker }
136*38e8c45fSAndroid Build Coastguard Worker 
RedirectStderr()137*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder& CommandOptions::CommandOptionsBuilder::RedirectStderr() {
138*38e8c45fSAndroid Build Coastguard Worker     values.output_mode_ = REDIRECT_TO_STDERR;
139*38e8c45fSAndroid Build Coastguard Worker     return *this;
140*38e8c45fSAndroid Build Coastguard Worker }
141*38e8c45fSAndroid Build Coastguard Worker 
142*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder&
CloseAllFileDescriptorsOnExec()143*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder::CloseAllFileDescriptorsOnExec() {
144*38e8c45fSAndroid Build Coastguard Worker     values.close_all_fds_on_exec_ = true;
145*38e8c45fSAndroid Build Coastguard Worker     return *this;
146*38e8c45fSAndroid Build Coastguard Worker }
147*38e8c45fSAndroid Build Coastguard Worker 
Log(const std::string & message)148*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder& CommandOptions::CommandOptionsBuilder::Log(
149*38e8c45fSAndroid Build Coastguard Worker     const std::string& message) {
150*38e8c45fSAndroid Build Coastguard Worker     values.logging_message_ = message;
151*38e8c45fSAndroid Build Coastguard Worker     return *this;
152*38e8c45fSAndroid Build Coastguard Worker }
153*38e8c45fSAndroid Build Coastguard Worker 
Build()154*38e8c45fSAndroid Build Coastguard Worker CommandOptions CommandOptions::CommandOptionsBuilder::Build() {
155*38e8c45fSAndroid Build Coastguard Worker     return CommandOptions(values);
156*38e8c45fSAndroid Build Coastguard Worker }
157*38e8c45fSAndroid Build Coastguard Worker 
CommandOptionsValues(int64_t timeout_ms)158*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsValues::CommandOptionsValues(int64_t timeout_ms)
159*38e8c45fSAndroid Build Coastguard Worker     : timeout_ms_(timeout_ms),
160*38e8c45fSAndroid Build Coastguard Worker       always_(false),
161*38e8c45fSAndroid Build Coastguard Worker       close_all_fds_on_exec_(false),
162*38e8c45fSAndroid Build Coastguard Worker       account_mode_(DONT_DROP_ROOT),
163*38e8c45fSAndroid Build Coastguard Worker       output_mode_(NORMAL_OUTPUT),
164*38e8c45fSAndroid Build Coastguard Worker       logging_message_("") {
165*38e8c45fSAndroid Build Coastguard Worker }
166*38e8c45fSAndroid Build Coastguard Worker 
CommandOptions(const CommandOptionsValues & values)167*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptions(const CommandOptionsValues& values) : values(values) {
168*38e8c45fSAndroid Build Coastguard Worker }
169*38e8c45fSAndroid Build Coastguard Worker 
Timeout() const170*38e8c45fSAndroid Build Coastguard Worker int64_t CommandOptions::Timeout() const {
171*38e8c45fSAndroid Build Coastguard Worker     return MSEC_TO_SEC(values.timeout_ms_);
172*38e8c45fSAndroid Build Coastguard Worker }
173*38e8c45fSAndroid Build Coastguard Worker 
TimeoutInMs() const174*38e8c45fSAndroid Build Coastguard Worker int64_t CommandOptions::TimeoutInMs() const {
175*38e8c45fSAndroid Build Coastguard Worker     return values.timeout_ms_;
176*38e8c45fSAndroid Build Coastguard Worker }
177*38e8c45fSAndroid Build Coastguard Worker 
Always() const178*38e8c45fSAndroid Build Coastguard Worker bool CommandOptions::Always() const {
179*38e8c45fSAndroid Build Coastguard Worker     return values.always_;
180*38e8c45fSAndroid Build Coastguard Worker }
181*38e8c45fSAndroid Build Coastguard Worker 
ShouldCloseAllFileDescriptorsOnExec() const182*38e8c45fSAndroid Build Coastguard Worker bool CommandOptions::ShouldCloseAllFileDescriptorsOnExec() const {
183*38e8c45fSAndroid Build Coastguard Worker     return values.close_all_fds_on_exec_;
184*38e8c45fSAndroid Build Coastguard Worker }
185*38e8c45fSAndroid Build Coastguard Worker 
PrivilegeMode() const186*38e8c45fSAndroid Build Coastguard Worker PrivilegeMode CommandOptions::PrivilegeMode() const {
187*38e8c45fSAndroid Build Coastguard Worker     return values.account_mode_;
188*38e8c45fSAndroid Build Coastguard Worker }
189*38e8c45fSAndroid Build Coastguard Worker 
OutputMode() const190*38e8c45fSAndroid Build Coastguard Worker OutputMode CommandOptions::OutputMode() const {
191*38e8c45fSAndroid Build Coastguard Worker     return values.output_mode_;
192*38e8c45fSAndroid Build Coastguard Worker }
193*38e8c45fSAndroid Build Coastguard Worker 
LoggingMessage() const194*38e8c45fSAndroid Build Coastguard Worker std::string CommandOptions::LoggingMessage() const {
195*38e8c45fSAndroid Build Coastguard Worker     return values.logging_message_;
196*38e8c45fSAndroid Build Coastguard Worker }
197*38e8c45fSAndroid Build Coastguard Worker 
WithTimeout(int64_t timeout_sec)198*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder CommandOptions::WithTimeout(int64_t timeout_sec) {
199*38e8c45fSAndroid Build Coastguard Worker     return CommandOptions::CommandOptionsBuilder(SEC_TO_MSEC(timeout_sec));
200*38e8c45fSAndroid Build Coastguard Worker }
201*38e8c45fSAndroid Build Coastguard Worker 
WithTimeoutInMs(int64_t timeout_ms)202*38e8c45fSAndroid Build Coastguard Worker CommandOptions::CommandOptionsBuilder CommandOptions::WithTimeoutInMs(int64_t timeout_ms) {
203*38e8c45fSAndroid Build Coastguard Worker     return CommandOptions::CommandOptionsBuilder(timeout_ms);
204*38e8c45fSAndroid Build Coastguard Worker }
205*38e8c45fSAndroid Build Coastguard Worker 
206*38e8c45fSAndroid Build Coastguard Worker std::string PropertiesHelper::build_type_ = "";
207*38e8c45fSAndroid Build Coastguard Worker int PropertiesHelper::dry_run_ = -1;
208*38e8c45fSAndroid Build Coastguard Worker int PropertiesHelper::unroot_ = -1;
209*38e8c45fSAndroid Build Coastguard Worker int PropertiesHelper::parallel_run_ = -1;
210*38e8c45fSAndroid Build Coastguard Worker int PropertiesHelper::strict_run_ = -1;
211*38e8c45fSAndroid Build Coastguard Worker 
IsUserBuild()212*38e8c45fSAndroid Build Coastguard Worker bool PropertiesHelper::IsUserBuild() {
213*38e8c45fSAndroid Build Coastguard Worker     if (build_type_.empty()) {
214*38e8c45fSAndroid Build Coastguard Worker         build_type_ = android::base::GetProperty("ro.build.type", "user");
215*38e8c45fSAndroid Build Coastguard Worker     }
216*38e8c45fSAndroid Build Coastguard Worker     return "user" == build_type_;
217*38e8c45fSAndroid Build Coastguard Worker }
218*38e8c45fSAndroid Build Coastguard Worker 
IsDryRun()219*38e8c45fSAndroid Build Coastguard Worker bool PropertiesHelper::IsDryRun() {
220*38e8c45fSAndroid Build Coastguard Worker     if (dry_run_ == -1) {
221*38e8c45fSAndroid Build Coastguard Worker         dry_run_ = android::base::GetBoolProperty("dumpstate.dry_run", false) ? 1 : 0;
222*38e8c45fSAndroid Build Coastguard Worker     }
223*38e8c45fSAndroid Build Coastguard Worker     return dry_run_ == 1;
224*38e8c45fSAndroid Build Coastguard Worker }
225*38e8c45fSAndroid Build Coastguard Worker 
IsUnroot()226*38e8c45fSAndroid Build Coastguard Worker bool PropertiesHelper::IsUnroot() {
227*38e8c45fSAndroid Build Coastguard Worker     if (unroot_ == -1) {
228*38e8c45fSAndroid Build Coastguard Worker         unroot_ = android::base::GetBoolProperty("dumpstate.unroot", false) ? 1 : 0;
229*38e8c45fSAndroid Build Coastguard Worker     }
230*38e8c45fSAndroid Build Coastguard Worker     return unroot_ == 1;
231*38e8c45fSAndroid Build Coastguard Worker }
232*38e8c45fSAndroid Build Coastguard Worker 
IsParallelRun()233*38e8c45fSAndroid Build Coastguard Worker bool PropertiesHelper::IsParallelRun() {
234*38e8c45fSAndroid Build Coastguard Worker     if (parallel_run_ == -1) {
235*38e8c45fSAndroid Build Coastguard Worker         parallel_run_ = android::base::GetBoolProperty("dumpstate.parallel_run",
236*38e8c45fSAndroid Build Coastguard Worker                 /* default_value = */true) ? 1 : 0;
237*38e8c45fSAndroid Build Coastguard Worker     }
238*38e8c45fSAndroid Build Coastguard Worker     return parallel_run_ == 1;
239*38e8c45fSAndroid Build Coastguard Worker }
240*38e8c45fSAndroid Build Coastguard Worker 
IsStrictRun()241*38e8c45fSAndroid Build Coastguard Worker bool PropertiesHelper::IsStrictRun() {
242*38e8c45fSAndroid Build Coastguard Worker     if (strict_run_ == -1) {
243*38e8c45fSAndroid Build Coastguard Worker         // Defaults to using stricter timeouts.
244*38e8c45fSAndroid Build Coastguard Worker         strict_run_ = android::base::GetBoolProperty("dumpstate.strict_run", true) ? 1 : 0;
245*38e8c45fSAndroid Build Coastguard Worker     }
246*38e8c45fSAndroid Build Coastguard Worker     return strict_run_ == 1;
247*38e8c45fSAndroid Build Coastguard Worker }
248*38e8c45fSAndroid Build Coastguard Worker 
DumpFileToFd(int out_fd,const std::string & title,const std::string & path)249*38e8c45fSAndroid Build Coastguard Worker int DumpFileToFd(int out_fd, const std::string& title, const std::string& path) {
250*38e8c45fSAndroid Build Coastguard Worker     android::base::unique_fd fd(TEMP_FAILURE_RETRY(open(path.c_str(), O_RDONLY | O_NONBLOCK | O_CLOEXEC)));
251*38e8c45fSAndroid Build Coastguard Worker     if (fd.get() < 0) {
252*38e8c45fSAndroid Build Coastguard Worker         int err = errno;
253*38e8c45fSAndroid Build Coastguard Worker         if (title.empty()) {
254*38e8c45fSAndroid Build Coastguard Worker             dprintf(out_fd, "*** Error dumping %s: %s\n", path.c_str(), strerror(err));
255*38e8c45fSAndroid Build Coastguard Worker         } else {
256*38e8c45fSAndroid Build Coastguard Worker             dprintf(out_fd, "*** Error dumping %s (%s): %s\n", path.c_str(), title.c_str(),
257*38e8c45fSAndroid Build Coastguard Worker                     strerror(err));
258*38e8c45fSAndroid Build Coastguard Worker         }
259*38e8c45fSAndroid Build Coastguard Worker         return -1;
260*38e8c45fSAndroid Build Coastguard Worker     }
261*38e8c45fSAndroid Build Coastguard Worker     return DumpFileFromFdToFd(title, path, fd.get(), out_fd, PropertiesHelper::IsDryRun());
262*38e8c45fSAndroid Build Coastguard Worker }
263*38e8c45fSAndroid Build Coastguard Worker 
RunCommandToFd(int fd,const std::string & title,const std::vector<std::string> & full_command,const CommandOptions & options)264*38e8c45fSAndroid Build Coastguard Worker int RunCommandToFd(int fd, const std::string& title, const std::vector<std::string>& full_command,
265*38e8c45fSAndroid Build Coastguard Worker                    const CommandOptions& options) {
266*38e8c45fSAndroid Build Coastguard Worker     if (full_command.empty()) {
267*38e8c45fSAndroid Build Coastguard Worker         MYLOGE("No arguments on RunCommandToFd(%s)\n", title.c_str());
268*38e8c45fSAndroid Build Coastguard Worker         return -1;
269*38e8c45fSAndroid Build Coastguard Worker     }
270*38e8c45fSAndroid Build Coastguard Worker 
271*38e8c45fSAndroid Build Coastguard Worker     int size = full_command.size() + 1;  // null terminated
272*38e8c45fSAndroid Build Coastguard Worker     int starting_index = 0;
273*38e8c45fSAndroid Build Coastguard Worker     if (options.PrivilegeMode() == SU_ROOT) {
274*38e8c45fSAndroid Build Coastguard Worker         starting_index = 2;  // "su" "root"
275*38e8c45fSAndroid Build Coastguard Worker         size += starting_index;
276*38e8c45fSAndroid Build Coastguard Worker     }
277*38e8c45fSAndroid Build Coastguard Worker 
278*38e8c45fSAndroid Build Coastguard Worker     std::vector<const char*> args;
279*38e8c45fSAndroid Build Coastguard Worker     args.resize(size);
280*38e8c45fSAndroid Build Coastguard Worker 
281*38e8c45fSAndroid Build Coastguard Worker     std::string command_string;
282*38e8c45fSAndroid Build Coastguard Worker     if (options.PrivilegeMode() == SU_ROOT) {
283*38e8c45fSAndroid Build Coastguard Worker         args[0] = kSuPath;
284*38e8c45fSAndroid Build Coastguard Worker         command_string += kSuPath;
285*38e8c45fSAndroid Build Coastguard Worker         args[1] = "root";
286*38e8c45fSAndroid Build Coastguard Worker         command_string += " root ";
287*38e8c45fSAndroid Build Coastguard Worker     }
288*38e8c45fSAndroid Build Coastguard Worker     for (size_t i = 0; i < full_command.size(); i++) {
289*38e8c45fSAndroid Build Coastguard Worker         args[i + starting_index] = full_command[i].data();
290*38e8c45fSAndroid Build Coastguard Worker         command_string += args[i + starting_index];
291*38e8c45fSAndroid Build Coastguard Worker         if (i != full_command.size() - 1) {
292*38e8c45fSAndroid Build Coastguard Worker             command_string += " ";
293*38e8c45fSAndroid Build Coastguard Worker         }
294*38e8c45fSAndroid Build Coastguard Worker     }
295*38e8c45fSAndroid Build Coastguard Worker     args[size - 1] = nullptr;
296*38e8c45fSAndroid Build Coastguard Worker 
297*38e8c45fSAndroid Build Coastguard Worker     const char* command = command_string.c_str();
298*38e8c45fSAndroid Build Coastguard Worker 
299*38e8c45fSAndroid Build Coastguard Worker     if (options.PrivilegeMode() == SU_ROOT && PropertiesHelper::IsUserBuild()) {
300*38e8c45fSAndroid Build Coastguard Worker         dprintf(fd, "Skipping '%s' on user build.\n", command);
301*38e8c45fSAndroid Build Coastguard Worker         return 0;
302*38e8c45fSAndroid Build Coastguard Worker     }
303*38e8c45fSAndroid Build Coastguard Worker 
304*38e8c45fSAndroid Build Coastguard Worker     if (!title.empty()) {
305*38e8c45fSAndroid Build Coastguard Worker         dprintf(fd, "------ %s (%s) ------\n", title.c_str(), command);
306*38e8c45fSAndroid Build Coastguard Worker     }
307*38e8c45fSAndroid Build Coastguard Worker 
308*38e8c45fSAndroid Build Coastguard Worker     const std::string& logging_message = options.LoggingMessage();
309*38e8c45fSAndroid Build Coastguard Worker     if (!logging_message.empty()) {
310*38e8c45fSAndroid Build Coastguard Worker         MYLOGI(logging_message.c_str(), command_string.c_str());
311*38e8c45fSAndroid Build Coastguard Worker     }
312*38e8c45fSAndroid Build Coastguard Worker 
313*38e8c45fSAndroid Build Coastguard Worker     bool silent = (options.OutputMode() == REDIRECT_TO_STDERR ||
314*38e8c45fSAndroid Build Coastguard Worker                    options.ShouldCloseAllFileDescriptorsOnExec());
315*38e8c45fSAndroid Build Coastguard Worker     bool redirecting_to_fd = STDOUT_FILENO != fd;
316*38e8c45fSAndroid Build Coastguard Worker 
317*38e8c45fSAndroid Build Coastguard Worker     if (PropertiesHelper::IsDryRun() && !options.Always()) {
318*38e8c45fSAndroid Build Coastguard Worker         if (!title.empty()) {
319*38e8c45fSAndroid Build Coastguard Worker             dprintf(fd, "\t(skipped on dry run)\n");
320*38e8c45fSAndroid Build Coastguard Worker         } else if (redirecting_to_fd) {
321*38e8c45fSAndroid Build Coastguard Worker             // There is no title, but we should still print a dry-run message
322*38e8c45fSAndroid Build Coastguard Worker             dprintf(fd, "%s: skipped on dry run\n", command_string.c_str());
323*38e8c45fSAndroid Build Coastguard Worker         }
324*38e8c45fSAndroid Build Coastguard Worker         return 0;
325*38e8c45fSAndroid Build Coastguard Worker     }
326*38e8c45fSAndroid Build Coastguard Worker 
327*38e8c45fSAndroid Build Coastguard Worker     const char* path = args[0];
328*38e8c45fSAndroid Build Coastguard Worker 
329*38e8c45fSAndroid Build Coastguard Worker     uint64_t start = Nanotime();
330*38e8c45fSAndroid Build Coastguard Worker     pid_t pid = vfork();
331*38e8c45fSAndroid Build Coastguard Worker 
332*38e8c45fSAndroid Build Coastguard Worker     /* handle error case */
333*38e8c45fSAndroid Build Coastguard Worker     if (pid < 0) {
334*38e8c45fSAndroid Build Coastguard Worker         if (!silent) dprintf(fd, "*** fork: %s\n", strerror(errno));
335*38e8c45fSAndroid Build Coastguard Worker         MYLOGE("*** fork: %s\n", strerror(errno));
336*38e8c45fSAndroid Build Coastguard Worker         return pid;
337*38e8c45fSAndroid Build Coastguard Worker     }
338*38e8c45fSAndroid Build Coastguard Worker 
339*38e8c45fSAndroid Build Coastguard Worker     /* handle child case */
340*38e8c45fSAndroid Build Coastguard Worker     if (pid == 0) {
341*38e8c45fSAndroid Build Coastguard Worker         if (options.PrivilegeMode() == DROP_ROOT && !DropRootUser()) {
342*38e8c45fSAndroid Build Coastguard Worker             if (!silent) {
343*38e8c45fSAndroid Build Coastguard Worker                 dprintf(fd, "*** failed to drop root before running %s: %s\n", command,
344*38e8c45fSAndroid Build Coastguard Worker                         strerror(errno));
345*38e8c45fSAndroid Build Coastguard Worker             }
346*38e8c45fSAndroid Build Coastguard Worker             MYLOGE("*** could not drop root before running %s: %s\n", command, strerror(errno));
347*38e8c45fSAndroid Build Coastguard Worker             _exit(EXIT_FAILURE);
348*38e8c45fSAndroid Build Coastguard Worker         }
349*38e8c45fSAndroid Build Coastguard Worker 
350*38e8c45fSAndroid Build Coastguard Worker         if (options.ShouldCloseAllFileDescriptorsOnExec()) {
351*38e8c45fSAndroid Build Coastguard Worker             int devnull_fd = TEMP_FAILURE_RETRY(open("/dev/null", O_RDONLY));
352*38e8c45fSAndroid Build Coastguard Worker             TEMP_FAILURE_RETRY(dup2(devnull_fd, STDIN_FILENO));
353*38e8c45fSAndroid Build Coastguard Worker             close(devnull_fd);
354*38e8c45fSAndroid Build Coastguard Worker             devnull_fd = TEMP_FAILURE_RETRY(open("/dev/null", O_WRONLY));
355*38e8c45fSAndroid Build Coastguard Worker             TEMP_FAILURE_RETRY(dup2(devnull_fd, STDOUT_FILENO));
356*38e8c45fSAndroid Build Coastguard Worker             TEMP_FAILURE_RETRY(dup2(devnull_fd, STDERR_FILENO));
357*38e8c45fSAndroid Build Coastguard Worker             close(devnull_fd);
358*38e8c45fSAndroid Build Coastguard Worker             // This is to avoid leaking FDs that, accidentally, have not been
359*38e8c45fSAndroid Build Coastguard Worker             // marked as O_CLOEXEC. Leaking FDs across exec can cause failures
360*38e8c45fSAndroid Build Coastguard Worker             // when execing a process that has a SELinux auto_trans rule.
361*38e8c45fSAndroid Build Coastguard Worker             // Here we assume that the dumpstate process didn't open more than
362*38e8c45fSAndroid Build Coastguard Worker             // 1000 FDs. In theory we could iterate through /proc/self/fd/, but
363*38e8c45fSAndroid Build Coastguard Worker             // doing that in a fork-safe way is too complex and not worth it
364*38e8c45fSAndroid Build Coastguard Worker             // (opendir()/readdir() do heap allocations and take locks).
365*38e8c45fSAndroid Build Coastguard Worker             for (int i = 0; i < 1000; i++) {
366*38e8c45fSAndroid Build Coastguard Worker                 if (i != STDIN_FILENO && i!= STDOUT_FILENO && i != STDERR_FILENO) {
367*38e8c45fSAndroid Build Coastguard Worker                     close(i);
368*38e8c45fSAndroid Build Coastguard Worker                 }
369*38e8c45fSAndroid Build Coastguard Worker             }
370*38e8c45fSAndroid Build Coastguard Worker         } else if (silent) {
371*38e8c45fSAndroid Build Coastguard Worker             // Redirects stdout to stderr
372*38e8c45fSAndroid Build Coastguard Worker             TEMP_FAILURE_RETRY(dup2(STDERR_FILENO, STDOUT_FILENO));
373*38e8c45fSAndroid Build Coastguard Worker         } else if (redirecting_to_fd) {
374*38e8c45fSAndroid Build Coastguard Worker             // Redirect stdout to fd
375*38e8c45fSAndroid Build Coastguard Worker             TEMP_FAILURE_RETRY(dup2(fd, STDOUT_FILENO));
376*38e8c45fSAndroid Build Coastguard Worker             close(fd);
377*38e8c45fSAndroid Build Coastguard Worker         }
378*38e8c45fSAndroid Build Coastguard Worker 
379*38e8c45fSAndroid Build Coastguard Worker         /* make sure the child dies when dumpstate dies */
380*38e8c45fSAndroid Build Coastguard Worker         prctl(PR_SET_PDEATHSIG, SIGKILL);
381*38e8c45fSAndroid Build Coastguard Worker 
382*38e8c45fSAndroid Build Coastguard Worker         /* just ignore SIGPIPE, will go down with parent's */
383*38e8c45fSAndroid Build Coastguard Worker         struct sigaction sigact;
384*38e8c45fSAndroid Build Coastguard Worker         memset(&sigact, 0, sizeof(sigact));
385*38e8c45fSAndroid Build Coastguard Worker         sigact.sa_handler = SIG_IGN;
386*38e8c45fSAndroid Build Coastguard Worker         sigaction(SIGPIPE, &sigact, nullptr);
387*38e8c45fSAndroid Build Coastguard Worker 
388*38e8c45fSAndroid Build Coastguard Worker         execvp(path, (char**)args.data());
389*38e8c45fSAndroid Build Coastguard Worker         // execvp's result will be handled after waitpid_with_timeout() below, but
390*38e8c45fSAndroid Build Coastguard Worker         // if it failed, it's safer to exit dumpstate.
391*38e8c45fSAndroid Build Coastguard Worker         MYLOGD("execvp on command '%s' failed (error: %s)\n", command, strerror(errno));
392*38e8c45fSAndroid Build Coastguard Worker         // Must call _exit (instead of exit), otherwise it will corrupt the zip
393*38e8c45fSAndroid Build Coastguard Worker         // file.
394*38e8c45fSAndroid Build Coastguard Worker         _exit(EXIT_FAILURE);
395*38e8c45fSAndroid Build Coastguard Worker     }
396*38e8c45fSAndroid Build Coastguard Worker 
397*38e8c45fSAndroid Build Coastguard Worker     /* handle parent case */
398*38e8c45fSAndroid Build Coastguard Worker     int status;
399*38e8c45fSAndroid Build Coastguard Worker     bool ret = waitpid_with_timeout(pid, options.TimeoutInMs(), &status);
400*38e8c45fSAndroid Build Coastguard Worker 
401*38e8c45fSAndroid Build Coastguard Worker     uint64_t elapsed = Nanotime() - start;
402*38e8c45fSAndroid Build Coastguard Worker     if (!ret) {
403*38e8c45fSAndroid Build Coastguard Worker         if (errno == ETIMEDOUT) {
404*38e8c45fSAndroid Build Coastguard Worker             if (!silent)
405*38e8c45fSAndroid Build Coastguard Worker                 dprintf(fd, "*** command '%s' timed out after %.3fs (killing pid %d)\n", command,
406*38e8c45fSAndroid Build Coastguard Worker                         static_cast<float>(elapsed) / NANOS_PER_SEC, pid);
407*38e8c45fSAndroid Build Coastguard Worker             MYLOGE("*** command '%s' timed out after %.3fs (killing pid %d)\n", command,
408*38e8c45fSAndroid Build Coastguard Worker                    static_cast<float>(elapsed) / NANOS_PER_SEC, pid);
409*38e8c45fSAndroid Build Coastguard Worker         } else {
410*38e8c45fSAndroid Build Coastguard Worker             if (!silent)
411*38e8c45fSAndroid Build Coastguard Worker                 dprintf(fd, "*** command '%s': Error after %.4fs (killing pid %d)\n", command,
412*38e8c45fSAndroid Build Coastguard Worker                         static_cast<float>(elapsed) / NANOS_PER_SEC, pid);
413*38e8c45fSAndroid Build Coastguard Worker             MYLOGE("command '%s': Error after %.4fs (killing pid %d)\n", command,
414*38e8c45fSAndroid Build Coastguard Worker                    static_cast<float>(elapsed) / NANOS_PER_SEC, pid);
415*38e8c45fSAndroid Build Coastguard Worker         }
416*38e8c45fSAndroid Build Coastguard Worker         kill(pid, SIGTERM);
417*38e8c45fSAndroid Build Coastguard Worker         if (!waitpid_with_timeout(pid, 5000, nullptr)) {
418*38e8c45fSAndroid Build Coastguard Worker             kill(pid, SIGKILL);
419*38e8c45fSAndroid Build Coastguard Worker             if (!waitpid_with_timeout(pid, 5000, nullptr)) {
420*38e8c45fSAndroid Build Coastguard Worker                 if (!silent)
421*38e8c45fSAndroid Build Coastguard Worker                     dprintf(fd, "could not kill command '%s' (pid %d) even with SIGKILL.\n",
422*38e8c45fSAndroid Build Coastguard Worker                             command, pid);
423*38e8c45fSAndroid Build Coastguard Worker                 MYLOGE("could not kill command '%s' (pid %d) even with SIGKILL.\n", command, pid);
424*38e8c45fSAndroid Build Coastguard Worker             }
425*38e8c45fSAndroid Build Coastguard Worker         }
426*38e8c45fSAndroid Build Coastguard Worker         return -1;
427*38e8c45fSAndroid Build Coastguard Worker     }
428*38e8c45fSAndroid Build Coastguard Worker 
429*38e8c45fSAndroid Build Coastguard Worker     if (WIFSIGNALED(status)) {
430*38e8c45fSAndroid Build Coastguard Worker         if (!silent)
431*38e8c45fSAndroid Build Coastguard Worker             dprintf(fd, "*** command '%s' failed: killed by signal %d\n", command, WTERMSIG(status));
432*38e8c45fSAndroid Build Coastguard Worker         MYLOGE("*** command '%s' failed: killed by signal %d\n", command, WTERMSIG(status));
433*38e8c45fSAndroid Build Coastguard Worker     } else if (WIFEXITED(status) && WEXITSTATUS(status) > 0) {
434*38e8c45fSAndroid Build Coastguard Worker         status = WEXITSTATUS(status);
435*38e8c45fSAndroid Build Coastguard Worker         if (!silent) dprintf(fd, "*** command '%s' failed: exit code %d\n", command, status);
436*38e8c45fSAndroid Build Coastguard Worker         MYLOGE("*** command '%s' failed: exit code %d\n", command, status);
437*38e8c45fSAndroid Build Coastguard Worker     }
438*38e8c45fSAndroid Build Coastguard Worker 
439*38e8c45fSAndroid Build Coastguard Worker     return status;
440*38e8c45fSAndroid Build Coastguard Worker }
441*38e8c45fSAndroid Build Coastguard Worker 
442*38e8c45fSAndroid Build Coastguard Worker }  // namespace dumpstate
443*38e8c45fSAndroid Build Coastguard Worker }  // namespace os
444*38e8c45fSAndroid Build Coastguard Worker }  // namespace android
445