1 /*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 #define DEBUG false
17 #include "Log.h"
18
19 #include "incidentd_util.h"
20
21 #include <sys/prctl.h>
22 #include <wait.h>
23
24 #include "section_list.h"
25
26 namespace android {
27 namespace os {
28 namespace incidentd {
29
30 using namespace android::base;
31
get_privacy_of_section(int id)32 const Privacy* get_privacy_of_section(int id) {
33 int l = 0;
34 int r = PRIVACY_POLICY_COUNT - 1;
35 while (l <= r) {
36 int mid = (l + r) >> 1;
37 const Privacy* p = PRIVACY_POLICY_LIST[mid];
38
39 if (p->field_id < (uint32_t)id) {
40 l = mid + 1;
41 } else if (p->field_id > (uint32_t)id) {
42 r = mid - 1;
43 } else {
44 return p;
45 }
46 }
47 return NULL;
48 }
49
50 // ================================================================================
Fpipe()51 Fpipe::Fpipe() : mRead(), mWrite() {}
52
~Fpipe()53 Fpipe::~Fpipe() { close(); }
54
close()55 bool Fpipe::close() {
56 mRead.reset();
57 mWrite.reset();
58 return true;
59 }
60
init()61 bool Fpipe::init() { return Pipe(&mRead, &mWrite); }
62
readFd()63 unique_fd& Fpipe::readFd() { return mRead; }
64
writeFd()65 unique_fd& Fpipe::writeFd() { return mWrite; }
66
fork_execute_cmd(char * const argv[],Fpipe * input,Fpipe * output)67 pid_t fork_execute_cmd(char* const argv[], Fpipe* input, Fpipe* output) {
68 // fork used in multithreaded environment, avoid adding unnecessary code in child process
69 pid_t pid = fork();
70 if (pid == 0) {
71 VLOG("[In child]cmd %s", argv[0]);
72 if (input != NULL && (TEMP_FAILURE_RETRY(dup2(input->readFd().get(), STDIN_FILENO)) < 0 ||
73 !input->close())) {
74 ALOGW("Failed to dup2 stdin.");
75 _exit(EXIT_FAILURE);
76 }
77 if (TEMP_FAILURE_RETRY(dup2(output->writeFd().get(), STDOUT_FILENO)) < 0 ||
78 !output->close()) {
79 ALOGW("Failed to dup2 stdout.");
80 _exit(EXIT_FAILURE);
81 }
82 /* make sure the child dies when incidentd dies */
83 prctl(PR_SET_PDEATHSIG, SIGKILL);
84 execvp(argv[0], argv);
85 _exit(errno); // always exits with failure if any
86 }
87 // close the fds used in child process.
88 if (input != NULL) input->readFd().reset();
89 output->writeFd().reset();
90 return pid;
91 }
92
93 // ================================================================================
varargs(const char * first,va_list rest)94 const char** varargs(const char* first, va_list rest) {
95 va_list copied_rest;
96 int numOfArgs = 1; // first is already count.
97
98 va_copy(copied_rest, rest);
99 while (va_arg(copied_rest, const char*) != NULL) {
100 numOfArgs++;
101 }
102 va_end(copied_rest);
103
104 // allocate extra 1 for NULL terminator
105 const char** ret = (const char**)malloc(sizeof(const char*) * (numOfArgs + 1));
106 ret[0] = first;
107 for (int i = 1; i < numOfArgs; i++) {
108 const char* arg = va_arg(rest, const char*);
109 ret[i] = arg;
110 }
111 ret[numOfArgs] = NULL;
112 return ret;
113 }
114
115 // ================================================================================
116 const uint64_t NANOS_PER_SEC = 1000000000;
Nanotime()117 uint64_t Nanotime() {
118 timespec ts;
119 clock_gettime(CLOCK_MONOTONIC, &ts);
120 return static_cast<uint64_t>(ts.tv_sec * NANOS_PER_SEC + ts.tv_nsec);
121 }
122
123 // ================================================================================
124 const int WAIT_MAX = 5;
125 const struct timespec WAIT_INTERVAL_NS = {0, 200 * 1000 * 1000};
126
statusCode(int status)127 static status_t statusCode(int status) {
128 if (WIFSIGNALED(status)) {
129 VLOG("return by signal: %s", strerror(WTERMSIG(status)));
130 return -WTERMSIG(status);
131 } else if (WIFEXITED(status) && WEXITSTATUS(status) > 0) {
132 VLOG("return by exit: %s", strerror(WEXITSTATUS(status)));
133 return -WEXITSTATUS(status);
134 }
135 return NO_ERROR;
136 }
137
kill_child(pid_t pid)138 status_t kill_child(pid_t pid) {
139 int status;
140 VLOG("try to kill child process %d", pid);
141 kill(pid, SIGKILL);
142 if (waitpid(pid, &status, 0) == -1) return -1;
143 return statusCode(status);
144 }
145
wait_child(pid_t pid)146 status_t wait_child(pid_t pid) {
147 int status;
148 bool died = false;
149 // wait for child to report status up to 1 seconds
150 for (int loop = 0; !died && loop < WAIT_MAX; loop++) {
151 if (waitpid(pid, &status, WNOHANG) == pid) died = true;
152 // sleep for 0.2 second
153 nanosleep(&WAIT_INTERVAL_NS, NULL);
154 }
155 if (!died) return kill_child(pid);
156 return statusCode(status);
157 }
158
159 } // namespace incidentd
160 } // namespace os
161 } // namespace android