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