1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker * Copyright (C) 2016 The Android Open Source Project
3*d57664e9SAndroid Build Coastguard Worker *
4*d57664e9SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*d57664e9SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*d57664e9SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*d57664e9SAndroid Build Coastguard Worker *
8*d57664e9SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*d57664e9SAndroid Build Coastguard Worker *
10*d57664e9SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*d57664e9SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*d57664e9SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*d57664e9SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*d57664e9SAndroid Build Coastguard Worker * limitations under the License.
15*d57664e9SAndroid Build Coastguard Worker */
16*d57664e9SAndroid Build Coastguard Worker
17*d57664e9SAndroid Build Coastguard Worker #define LOG_TAG "incident"
18*d57664e9SAndroid Build Coastguard Worker
19*d57664e9SAndroid Build Coastguard Worker #include "incident_sections.h"
20*d57664e9SAndroid Build Coastguard Worker
21*d57664e9SAndroid Build Coastguard Worker #include <android/os/BnIncidentReportStatusListener.h>
22*d57664e9SAndroid Build Coastguard Worker #include <android/os/IIncidentManager.h>
23*d57664e9SAndroid Build Coastguard Worker #include <android/os/IncidentReportArgs.h>
24*d57664e9SAndroid Build Coastguard Worker #include <android/util/ProtoOutputStream.h>
25*d57664e9SAndroid Build Coastguard Worker #include <binder/IPCThreadState.h>
26*d57664e9SAndroid Build Coastguard Worker #include <binder/IServiceManager.h>
27*d57664e9SAndroid Build Coastguard Worker #include <utils/Looper.h>
28*d57664e9SAndroid Build Coastguard Worker
29*d57664e9SAndroid Build Coastguard Worker #include <cstring>
30*d57664e9SAndroid Build Coastguard Worker #include <fcntl.h>
31*d57664e9SAndroid Build Coastguard Worker #include <getopt.h>
32*d57664e9SAndroid Build Coastguard Worker #include <stdio.h>
33*d57664e9SAndroid Build Coastguard Worker #include <stdlib.h>
34*d57664e9SAndroid Build Coastguard Worker #include <unistd.h>
35*d57664e9SAndroid Build Coastguard Worker
36*d57664e9SAndroid Build Coastguard Worker using namespace android;
37*d57664e9SAndroid Build Coastguard Worker using namespace android::base;
38*d57664e9SAndroid Build Coastguard Worker using namespace android::binder;
39*d57664e9SAndroid Build Coastguard Worker using namespace android::os;
40*d57664e9SAndroid Build Coastguard Worker using android::util::FIELD_COUNT_SINGLE;
41*d57664e9SAndroid Build Coastguard Worker using android::util::FIELD_TYPE_STRING;
42*d57664e9SAndroid Build Coastguard Worker using android::util::ProtoOutputStream;
43*d57664e9SAndroid Build Coastguard Worker
44*d57664e9SAndroid Build Coastguard Worker // ================================================================================
45*d57664e9SAndroid Build Coastguard Worker class StatusListener : public BnIncidentReportStatusListener {
46*d57664e9SAndroid Build Coastguard Worker public:
47*d57664e9SAndroid Build Coastguard Worker StatusListener();
48*d57664e9SAndroid Build Coastguard Worker virtual ~StatusListener();
49*d57664e9SAndroid Build Coastguard Worker
50*d57664e9SAndroid Build Coastguard Worker virtual Status onReportStarted();
51*d57664e9SAndroid Build Coastguard Worker virtual Status onReportSectionStatus(int32_t section, int32_t status);
52*d57664e9SAndroid Build Coastguard Worker virtual Status onReportServiceStatus(const String16& service, int32_t status);
53*d57664e9SAndroid Build Coastguard Worker virtual Status onReportFinished();
54*d57664e9SAndroid Build Coastguard Worker virtual Status onReportFailed();
55*d57664e9SAndroid Build Coastguard Worker
56*d57664e9SAndroid Build Coastguard Worker int getExitCodeOrElse(int defaultCode);
57*d57664e9SAndroid Build Coastguard Worker private:
58*d57664e9SAndroid Build Coastguard Worker int mExitCode;
59*d57664e9SAndroid Build Coastguard Worker };
60*d57664e9SAndroid Build Coastguard Worker
StatusListener()61*d57664e9SAndroid Build Coastguard Worker StatusListener::StatusListener(): mExitCode(-1)
62*d57664e9SAndroid Build Coastguard Worker {
63*d57664e9SAndroid Build Coastguard Worker }
64*d57664e9SAndroid Build Coastguard Worker
~StatusListener()65*d57664e9SAndroid Build Coastguard Worker StatusListener::~StatusListener()
66*d57664e9SAndroid Build Coastguard Worker {
67*d57664e9SAndroid Build Coastguard Worker }
68*d57664e9SAndroid Build Coastguard Worker
69*d57664e9SAndroid Build Coastguard Worker Status
onReportStarted()70*d57664e9SAndroid Build Coastguard Worker StatusListener::onReportStarted()
71*d57664e9SAndroid Build Coastguard Worker {
72*d57664e9SAndroid Build Coastguard Worker return Status::ok();
73*d57664e9SAndroid Build Coastguard Worker }
74*d57664e9SAndroid Build Coastguard Worker
75*d57664e9SAndroid Build Coastguard Worker Status
onReportSectionStatus(int32_t section,int32_t status)76*d57664e9SAndroid Build Coastguard Worker StatusListener::onReportSectionStatus(int32_t section, int32_t status)
77*d57664e9SAndroid Build Coastguard Worker {
78*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "section %d status %d\n", section, status);
79*d57664e9SAndroid Build Coastguard Worker ALOGD("section %d status %d\n", section, status);
80*d57664e9SAndroid Build Coastguard Worker return Status::ok();
81*d57664e9SAndroid Build Coastguard Worker }
82*d57664e9SAndroid Build Coastguard Worker
83*d57664e9SAndroid Build Coastguard Worker Status
onReportServiceStatus(const String16 & service,int32_t status)84*d57664e9SAndroid Build Coastguard Worker StatusListener::onReportServiceStatus(const String16& service, int32_t status)
85*d57664e9SAndroid Build Coastguard Worker {
86*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "service '%s' status %d\n", String8(service).c_str(), status);
87*d57664e9SAndroid Build Coastguard Worker ALOGD("service '%s' status %d\n", String8(service).c_str(), status);
88*d57664e9SAndroid Build Coastguard Worker return Status::ok();
89*d57664e9SAndroid Build Coastguard Worker }
90*d57664e9SAndroid Build Coastguard Worker
91*d57664e9SAndroid Build Coastguard Worker Status
onReportFinished()92*d57664e9SAndroid Build Coastguard Worker StatusListener::onReportFinished()
93*d57664e9SAndroid Build Coastguard Worker {
94*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "done\n");
95*d57664e9SAndroid Build Coastguard Worker ALOGD("done\n");
96*d57664e9SAndroid Build Coastguard Worker mExitCode = 0;
97*d57664e9SAndroid Build Coastguard Worker return Status::ok();
98*d57664e9SAndroid Build Coastguard Worker }
99*d57664e9SAndroid Build Coastguard Worker
100*d57664e9SAndroid Build Coastguard Worker Status
onReportFailed()101*d57664e9SAndroid Build Coastguard Worker StatusListener::onReportFailed()
102*d57664e9SAndroid Build Coastguard Worker {
103*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "failed\n");
104*d57664e9SAndroid Build Coastguard Worker ALOGD("failed\n");
105*d57664e9SAndroid Build Coastguard Worker mExitCode = 1;
106*d57664e9SAndroid Build Coastguard Worker return Status::ok();
107*d57664e9SAndroid Build Coastguard Worker }
108*d57664e9SAndroid Build Coastguard Worker
109*d57664e9SAndroid Build Coastguard Worker int
getExitCodeOrElse(int defaultCode)110*d57664e9SAndroid Build Coastguard Worker StatusListener::getExitCodeOrElse(int defaultCode) {
111*d57664e9SAndroid Build Coastguard Worker return mExitCode == -1 ? defaultCode : mExitCode;
112*d57664e9SAndroid Build Coastguard Worker }
113*d57664e9SAndroid Build Coastguard Worker
114*d57664e9SAndroid Build Coastguard Worker // ================================================================================
section_list(FILE * out)115*d57664e9SAndroid Build Coastguard Worker static void section_list(FILE* out) {
116*d57664e9SAndroid Build Coastguard Worker IncidentSection sections[INCIDENT_SECTION_COUNT];
117*d57664e9SAndroid Build Coastguard Worker int i = 0;
118*d57664e9SAndroid Build Coastguard Worker int j = 0;
119*d57664e9SAndroid Build Coastguard Worker // sort the sections based on id
120*d57664e9SAndroid Build Coastguard Worker while (i < INCIDENT_SECTION_COUNT) {
121*d57664e9SAndroid Build Coastguard Worker IncidentSection curr = INCIDENT_SECTIONS[i];
122*d57664e9SAndroid Build Coastguard Worker for (int k = 0; k < j; k++) {
123*d57664e9SAndroid Build Coastguard Worker if (curr.id > sections[k].id) {
124*d57664e9SAndroid Build Coastguard Worker continue;
125*d57664e9SAndroid Build Coastguard Worker }
126*d57664e9SAndroid Build Coastguard Worker IncidentSection tmp = curr;
127*d57664e9SAndroid Build Coastguard Worker curr = sections[k];
128*d57664e9SAndroid Build Coastguard Worker sections[k] = tmp;
129*d57664e9SAndroid Build Coastguard Worker }
130*d57664e9SAndroid Build Coastguard Worker sections[j] = curr;
131*d57664e9SAndroid Build Coastguard Worker i++;
132*d57664e9SAndroid Build Coastguard Worker j++;
133*d57664e9SAndroid Build Coastguard Worker }
134*d57664e9SAndroid Build Coastguard Worker
135*d57664e9SAndroid Build Coastguard Worker fprintf(out, "available sections:\n");
136*d57664e9SAndroid Build Coastguard Worker for (int i = 0; i < INCIDENT_SECTION_COUNT; ++i) {
137*d57664e9SAndroid Build Coastguard Worker fprintf(out, "id: %4d, name: %s\n", sections[i].id, sections[i].name);
138*d57664e9SAndroid Build Coastguard Worker }
139*d57664e9SAndroid Build Coastguard Worker }
140*d57664e9SAndroid Build Coastguard Worker
141*d57664e9SAndroid Build Coastguard Worker // ================================================================================
142*d57664e9SAndroid Build Coastguard Worker static IncidentSection const*
find_section(const char * name)143*d57664e9SAndroid Build Coastguard Worker find_section(const char* name)
144*d57664e9SAndroid Build Coastguard Worker {
145*d57664e9SAndroid Build Coastguard Worker ssize_t low = 0;
146*d57664e9SAndroid Build Coastguard Worker ssize_t high = INCIDENT_SECTION_COUNT - 1;
147*d57664e9SAndroid Build Coastguard Worker
148*d57664e9SAndroid Build Coastguard Worker while (low <= high) {
149*d57664e9SAndroid Build Coastguard Worker ssize_t mid = (low + high) / 2;
150*d57664e9SAndroid Build Coastguard Worker IncidentSection const* section = INCIDENT_SECTIONS + mid;
151*d57664e9SAndroid Build Coastguard Worker
152*d57664e9SAndroid Build Coastguard Worker int cmp = strcmp(section->name, name);
153*d57664e9SAndroid Build Coastguard Worker if (cmp < 0) {
154*d57664e9SAndroid Build Coastguard Worker low = mid + 1;
155*d57664e9SAndroid Build Coastguard Worker } else if (cmp > 0) {
156*d57664e9SAndroid Build Coastguard Worker high = mid - 1;
157*d57664e9SAndroid Build Coastguard Worker } else {
158*d57664e9SAndroid Build Coastguard Worker return section;
159*d57664e9SAndroid Build Coastguard Worker }
160*d57664e9SAndroid Build Coastguard Worker }
161*d57664e9SAndroid Build Coastguard Worker return NULL;
162*d57664e9SAndroid Build Coastguard Worker }
163*d57664e9SAndroid Build Coastguard Worker
164*d57664e9SAndroid Build Coastguard Worker // ================================================================================
165*d57664e9SAndroid Build Coastguard Worker static int
get_privacy_policy(const char * arg)166*d57664e9SAndroid Build Coastguard Worker get_privacy_policy(const char* arg)
167*d57664e9SAndroid Build Coastguard Worker {
168*d57664e9SAndroid Build Coastguard Worker if (strcmp(arg, "L") == 0
169*d57664e9SAndroid Build Coastguard Worker || strcmp(arg, "LOCAL") == 0) {
170*d57664e9SAndroid Build Coastguard Worker return PRIVACY_POLICY_LOCAL;
171*d57664e9SAndroid Build Coastguard Worker }
172*d57664e9SAndroid Build Coastguard Worker if (strcmp(arg, "E") == 0
173*d57664e9SAndroid Build Coastguard Worker || strcmp(arg, "EXPLICIT") == 0) {
174*d57664e9SAndroid Build Coastguard Worker return PRIVACY_POLICY_EXPLICIT;
175*d57664e9SAndroid Build Coastguard Worker }
176*d57664e9SAndroid Build Coastguard Worker if (strcmp(arg, "A") == 0
177*d57664e9SAndroid Build Coastguard Worker || strcmp(arg, "AUTO") == 0
178*d57664e9SAndroid Build Coastguard Worker || strcmp(arg, "AUTOMATIC") == 0) {
179*d57664e9SAndroid Build Coastguard Worker return PRIVACY_POLICY_AUTOMATIC;
180*d57664e9SAndroid Build Coastguard Worker }
181*d57664e9SAndroid Build Coastguard Worker return -1; // return the default value
182*d57664e9SAndroid Build Coastguard Worker }
183*d57664e9SAndroid Build Coastguard Worker
184*d57664e9SAndroid Build Coastguard Worker // ================================================================================
185*d57664e9SAndroid Build Coastguard Worker static bool
parse_receiver_arg(const string & arg,string * pkg,string * cls)186*d57664e9SAndroid Build Coastguard Worker parse_receiver_arg(const string& arg, string* pkg, string* cls)
187*d57664e9SAndroid Build Coastguard Worker {
188*d57664e9SAndroid Build Coastguard Worker if (arg.length() == 0) {
189*d57664e9SAndroid Build Coastguard Worker return true;
190*d57664e9SAndroid Build Coastguard Worker }
191*d57664e9SAndroid Build Coastguard Worker size_t slash = arg.find('/');
192*d57664e9SAndroid Build Coastguard Worker if (slash == string::npos) {
193*d57664e9SAndroid Build Coastguard Worker return false;
194*d57664e9SAndroid Build Coastguard Worker }
195*d57664e9SAndroid Build Coastguard Worker if (slash == 0 || slash == arg.length() - 1) {
196*d57664e9SAndroid Build Coastguard Worker return false;
197*d57664e9SAndroid Build Coastguard Worker }
198*d57664e9SAndroid Build Coastguard Worker if (arg.find('/', slash+1) != string::npos) {
199*d57664e9SAndroid Build Coastguard Worker return false;
200*d57664e9SAndroid Build Coastguard Worker }
201*d57664e9SAndroid Build Coastguard Worker pkg->assign(arg, 0, slash);
202*d57664e9SAndroid Build Coastguard Worker cls->assign(arg, slash+1);
203*d57664e9SAndroid Build Coastguard Worker if ((*cls)[0] == '.') {
204*d57664e9SAndroid Build Coastguard Worker *cls = (*pkg) + (*cls);
205*d57664e9SAndroid Build Coastguard Worker }
206*d57664e9SAndroid Build Coastguard Worker return true;
207*d57664e9SAndroid Build Coastguard Worker }
208*d57664e9SAndroid Build Coastguard Worker
209*d57664e9SAndroid Build Coastguard Worker // ================================================================================
210*d57664e9SAndroid Build Coastguard Worker static int
stream_output(const int read_fd,const int write_fd)211*d57664e9SAndroid Build Coastguard Worker stream_output(const int read_fd, const int write_fd) {
212*d57664e9SAndroid Build Coastguard Worker while (true) {
213*d57664e9SAndroid Build Coastguard Worker int amt = splice(read_fd, NULL, write_fd, NULL, 4096, 0);
214*d57664e9SAndroid Build Coastguard Worker if (amt < 0) {
215*d57664e9SAndroid Build Coastguard Worker return errno;
216*d57664e9SAndroid Build Coastguard Worker } else if (amt == 0) {
217*d57664e9SAndroid Build Coastguard Worker return 0;
218*d57664e9SAndroid Build Coastguard Worker }
219*d57664e9SAndroid Build Coastguard Worker }
220*d57664e9SAndroid Build Coastguard Worker }
221*d57664e9SAndroid Build Coastguard Worker
222*d57664e9SAndroid Build Coastguard Worker // ================================================================================
223*d57664e9SAndroid Build Coastguard Worker static void
usage(FILE * out)224*d57664e9SAndroid Build Coastguard Worker usage(FILE* out)
225*d57664e9SAndroid Build Coastguard Worker {
226*d57664e9SAndroid Build Coastguard Worker fprintf(out, "usage: incident OPTIONS [SECTION...]\n");
227*d57664e9SAndroid Build Coastguard Worker fprintf(out, "\n");
228*d57664e9SAndroid Build Coastguard Worker fprintf(out, "Takes an incident report.\n");
229*d57664e9SAndroid Build Coastguard Worker fprintf(out, "\n");
230*d57664e9SAndroid Build Coastguard Worker fprintf(out, "OPTIONS\n");
231*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -l list available sections\n");
232*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -p privacy spec, LOCAL, EXPLICIT or AUTOMATIC. Default AUTOMATIC.\n");
233*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -r REASON human readable description of why the report is taken.\n");
234*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -z gzip the incident report, i.e. pipe the output through gzip.\n");
235*d57664e9SAndroid Build Coastguard Worker fprintf(out, "\n");
236*d57664e9SAndroid Build Coastguard Worker fprintf(out, "and one of these destinations:\n");
237*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -b (default) print the report to stdout (in proto format)\n");
238*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -d send the report into dropbox\n");
239*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -u print a full report to stdout for dumpstate to zip as a bug\n");
240*d57664e9SAndroid Build Coastguard Worker fprintf(out, " report. SECTION is ignored. Should only be called by dumpstate.\n");
241*d57664e9SAndroid Build Coastguard Worker fprintf(out, " -s PKG/CLS send broadcast to the broadcast receiver.\n");
242*d57664e9SAndroid Build Coastguard Worker fprintf(out, "\n");
243*d57664e9SAndroid Build Coastguard Worker fprintf(out, " SECTION the field numbers of the incident report fields to include\n");
244*d57664e9SAndroid Build Coastguard Worker fprintf(out, "\n");
245*d57664e9SAndroid Build Coastguard Worker }
246*d57664e9SAndroid Build Coastguard Worker
247*d57664e9SAndroid Build Coastguard Worker int
main(int argc,char ** argv)248*d57664e9SAndroid Build Coastguard Worker main(int argc, char** argv)
249*d57664e9SAndroid Build Coastguard Worker {
250*d57664e9SAndroid Build Coastguard Worker Status status;
251*d57664e9SAndroid Build Coastguard Worker IncidentReportArgs args;
252*d57664e9SAndroid Build Coastguard Worker enum { DEST_UNSET, DEST_DROPBOX, DEST_STDOUT, DEST_BROADCAST, DEST_DUMPSTATE } destination = DEST_UNSET;
253*d57664e9SAndroid Build Coastguard Worker int privacyPolicy = PRIVACY_POLICY_AUTOMATIC;
254*d57664e9SAndroid Build Coastguard Worker string reason;
255*d57664e9SAndroid Build Coastguard Worker string receiverArg;
256*d57664e9SAndroid Build Coastguard Worker
257*d57664e9SAndroid Build Coastguard Worker // Parse the args
258*d57664e9SAndroid Build Coastguard Worker int opt;
259*d57664e9SAndroid Build Coastguard Worker while ((opt = getopt(argc, argv, "bhdlp:r:s:uz")) != -1) {
260*d57664e9SAndroid Build Coastguard Worker switch (opt) {
261*d57664e9SAndroid Build Coastguard Worker case 'h':
262*d57664e9SAndroid Build Coastguard Worker usage(stdout);
263*d57664e9SAndroid Build Coastguard Worker return 0;
264*d57664e9SAndroid Build Coastguard Worker case 'l':
265*d57664e9SAndroid Build Coastguard Worker section_list(stdout);
266*d57664e9SAndroid Build Coastguard Worker return 0;
267*d57664e9SAndroid Build Coastguard Worker case 'b':
268*d57664e9SAndroid Build Coastguard Worker if (!(destination == DEST_UNSET || destination == DEST_STDOUT)) {
269*d57664e9SAndroid Build Coastguard Worker usage(stderr);
270*d57664e9SAndroid Build Coastguard Worker return 1;
271*d57664e9SAndroid Build Coastguard Worker }
272*d57664e9SAndroid Build Coastguard Worker destination = DEST_STDOUT;
273*d57664e9SAndroid Build Coastguard Worker break;
274*d57664e9SAndroid Build Coastguard Worker case 'd':
275*d57664e9SAndroid Build Coastguard Worker if (!(destination == DEST_UNSET || destination == DEST_DROPBOX)) {
276*d57664e9SAndroid Build Coastguard Worker usage(stderr);
277*d57664e9SAndroid Build Coastguard Worker return 1;
278*d57664e9SAndroid Build Coastguard Worker }
279*d57664e9SAndroid Build Coastguard Worker destination = DEST_DROPBOX;
280*d57664e9SAndroid Build Coastguard Worker break;
281*d57664e9SAndroid Build Coastguard Worker case 'u':
282*d57664e9SAndroid Build Coastguard Worker if (!(destination == DEST_UNSET || destination == DEST_DUMPSTATE)) {
283*d57664e9SAndroid Build Coastguard Worker usage(stderr);
284*d57664e9SAndroid Build Coastguard Worker return 1;
285*d57664e9SAndroid Build Coastguard Worker }
286*d57664e9SAndroid Build Coastguard Worker destination = DEST_DUMPSTATE;
287*d57664e9SAndroid Build Coastguard Worker break;
288*d57664e9SAndroid Build Coastguard Worker case 'p':
289*d57664e9SAndroid Build Coastguard Worker privacyPolicy = get_privacy_policy(optarg);
290*d57664e9SAndroid Build Coastguard Worker break;
291*d57664e9SAndroid Build Coastguard Worker case 'r':
292*d57664e9SAndroid Build Coastguard Worker if (reason.size() > 0) {
293*d57664e9SAndroid Build Coastguard Worker usage(stderr);
294*d57664e9SAndroid Build Coastguard Worker return 1;
295*d57664e9SAndroid Build Coastguard Worker }
296*d57664e9SAndroid Build Coastguard Worker reason = optarg;
297*d57664e9SAndroid Build Coastguard Worker break;
298*d57664e9SAndroid Build Coastguard Worker case 's':
299*d57664e9SAndroid Build Coastguard Worker if (destination != DEST_UNSET) {
300*d57664e9SAndroid Build Coastguard Worker usage(stderr);
301*d57664e9SAndroid Build Coastguard Worker return 1;
302*d57664e9SAndroid Build Coastguard Worker }
303*d57664e9SAndroid Build Coastguard Worker destination = DEST_BROADCAST;
304*d57664e9SAndroid Build Coastguard Worker receiverArg = optarg;
305*d57664e9SAndroid Build Coastguard Worker break;
306*d57664e9SAndroid Build Coastguard Worker case 'z':
307*d57664e9SAndroid Build Coastguard Worker args.setGzip(true);
308*d57664e9SAndroid Build Coastguard Worker break;
309*d57664e9SAndroid Build Coastguard Worker default:
310*d57664e9SAndroid Build Coastguard Worker usage(stderr);
311*d57664e9SAndroid Build Coastguard Worker return 1;
312*d57664e9SAndroid Build Coastguard Worker }
313*d57664e9SAndroid Build Coastguard Worker }
314*d57664e9SAndroid Build Coastguard Worker if (destination == DEST_UNSET) {
315*d57664e9SAndroid Build Coastguard Worker destination = DEST_STDOUT;
316*d57664e9SAndroid Build Coastguard Worker }
317*d57664e9SAndroid Build Coastguard Worker
318*d57664e9SAndroid Build Coastguard Worker string pkg;
319*d57664e9SAndroid Build Coastguard Worker string cls;
320*d57664e9SAndroid Build Coastguard Worker if (parse_receiver_arg(receiverArg, &pkg, &cls)) {
321*d57664e9SAndroid Build Coastguard Worker args.setReceiverPkg(pkg);
322*d57664e9SAndroid Build Coastguard Worker args.setReceiverCls(cls);
323*d57664e9SAndroid Build Coastguard Worker } else {
324*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "badly formatted -s package/class option: %s\n\n", receiverArg.c_str());
325*d57664e9SAndroid Build Coastguard Worker usage(stderr);
326*d57664e9SAndroid Build Coastguard Worker return 1;
327*d57664e9SAndroid Build Coastguard Worker }
328*d57664e9SAndroid Build Coastguard Worker
329*d57664e9SAndroid Build Coastguard Worker if (optind == argc) {
330*d57664e9SAndroid Build Coastguard Worker args.setAll(true);
331*d57664e9SAndroid Build Coastguard Worker } else {
332*d57664e9SAndroid Build Coastguard Worker for (int i=optind; i<argc; i++) {
333*d57664e9SAndroid Build Coastguard Worker const char* arg = argv[i];
334*d57664e9SAndroid Build Coastguard Worker char* end;
335*d57664e9SAndroid Build Coastguard Worker if (arg[0] != '\0') {
336*d57664e9SAndroid Build Coastguard Worker int section = strtol(arg, &end, 0);
337*d57664e9SAndroid Build Coastguard Worker if (*end == '\0') {
338*d57664e9SAndroid Build Coastguard Worker args.addSection(section);
339*d57664e9SAndroid Build Coastguard Worker } else {
340*d57664e9SAndroid Build Coastguard Worker IncidentSection const* ic = find_section(arg);
341*d57664e9SAndroid Build Coastguard Worker if (ic == NULL) {
342*d57664e9SAndroid Build Coastguard Worker ALOGD("Invalid section: %s\n", arg);
343*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "Invalid section: %s\n", arg);
344*d57664e9SAndroid Build Coastguard Worker return 1;
345*d57664e9SAndroid Build Coastguard Worker }
346*d57664e9SAndroid Build Coastguard Worker args.addSection(ic->id);
347*d57664e9SAndroid Build Coastguard Worker }
348*d57664e9SAndroid Build Coastguard Worker }
349*d57664e9SAndroid Build Coastguard Worker }
350*d57664e9SAndroid Build Coastguard Worker }
351*d57664e9SAndroid Build Coastguard Worker args.setPrivacyPolicy(privacyPolicy);
352*d57664e9SAndroid Build Coastguard Worker
353*d57664e9SAndroid Build Coastguard Worker if (reason.size() > 0) {
354*d57664e9SAndroid Build Coastguard Worker ProtoOutputStream proto;
355*d57664e9SAndroid Build Coastguard Worker proto.write(/* reason field id */ 2 | FIELD_TYPE_STRING | FIELD_COUNT_SINGLE, reason);
356*d57664e9SAndroid Build Coastguard Worker vector<uint8_t> header;
357*d57664e9SAndroid Build Coastguard Worker proto.serializeToVector(&header);
358*d57664e9SAndroid Build Coastguard Worker args.addHeader(header);
359*d57664e9SAndroid Build Coastguard Worker }
360*d57664e9SAndroid Build Coastguard Worker
361*d57664e9SAndroid Build Coastguard Worker // Start the thread pool.
362*d57664e9SAndroid Build Coastguard Worker sp<ProcessState> ps(ProcessState::self());
363*d57664e9SAndroid Build Coastguard Worker ps->startThreadPool();
364*d57664e9SAndroid Build Coastguard Worker ps->giveThreadPoolName();
365*d57664e9SAndroid Build Coastguard Worker
366*d57664e9SAndroid Build Coastguard Worker // Look up the service
367*d57664e9SAndroid Build Coastguard Worker sp<IIncidentManager> service = interface_cast<IIncidentManager>(
368*d57664e9SAndroid Build Coastguard Worker defaultServiceManager()->getService(android::String16("incident")));
369*d57664e9SAndroid Build Coastguard Worker if (service == NULL) {
370*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "Couldn't look up the incident service\n");
371*d57664e9SAndroid Build Coastguard Worker return 1;
372*d57664e9SAndroid Build Coastguard Worker }
373*d57664e9SAndroid Build Coastguard Worker
374*d57664e9SAndroid Build Coastguard Worker // Construct the stream
375*d57664e9SAndroid Build Coastguard Worker int fds[2];
376*d57664e9SAndroid Build Coastguard Worker pipe(fds);
377*d57664e9SAndroid Build Coastguard Worker
378*d57664e9SAndroid Build Coastguard Worker unique_fd readEnd(fds[0]);
379*d57664e9SAndroid Build Coastguard Worker unique_fd writeEnd(fds[1]);
380*d57664e9SAndroid Build Coastguard Worker
381*d57664e9SAndroid Build Coastguard Worker if (destination == DEST_STDOUT) {
382*d57664e9SAndroid Build Coastguard Worker // Call into the service
383*d57664e9SAndroid Build Coastguard Worker sp<StatusListener> listener(new StatusListener());
384*d57664e9SAndroid Build Coastguard Worker status = service->reportIncidentToStream(args, listener, std::move(writeEnd));
385*d57664e9SAndroid Build Coastguard Worker
386*d57664e9SAndroid Build Coastguard Worker if (!status.isOk()) {
387*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "reportIncident returned \"%s\"\n", status.toString8().c_str());
388*d57664e9SAndroid Build Coastguard Worker return 1;
389*d57664e9SAndroid Build Coastguard Worker }
390*d57664e9SAndroid Build Coastguard Worker
391*d57664e9SAndroid Build Coastguard Worker // Wait for the result and print out the data they send.
392*d57664e9SAndroid Build Coastguard Worker //IPCThreadState::self()->joinThreadPool();
393*d57664e9SAndroid Build Coastguard Worker return listener->getExitCodeOrElse(stream_output(fds[0], STDOUT_FILENO));
394*d57664e9SAndroid Build Coastguard Worker } else if (destination == DEST_DUMPSTATE) {
395*d57664e9SAndroid Build Coastguard Worker // Call into the service
396*d57664e9SAndroid Build Coastguard Worker sp<StatusListener> listener(new StatusListener());
397*d57664e9SAndroid Build Coastguard Worker status = service->reportIncidentToDumpstate(std::move(writeEnd), listener);
398*d57664e9SAndroid Build Coastguard Worker if (!status.isOk()) {
399*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "reportIncident returned \"%s\"\n", status.toString8().c_str());
400*d57664e9SAndroid Build Coastguard Worker return 1;
401*d57664e9SAndroid Build Coastguard Worker }
402*d57664e9SAndroid Build Coastguard Worker return listener->getExitCodeOrElse(stream_output(fds[0], STDOUT_FILENO));
403*d57664e9SAndroid Build Coastguard Worker } else {
404*d57664e9SAndroid Build Coastguard Worker status = service->reportIncident(args);
405*d57664e9SAndroid Build Coastguard Worker if (!status.isOk()) {
406*d57664e9SAndroid Build Coastguard Worker fprintf(stderr, "reportIncident returned \"%s\"\n", status.toString8().c_str());
407*d57664e9SAndroid Build Coastguard Worker return 1;
408*d57664e9SAndroid Build Coastguard Worker } else {
409*d57664e9SAndroid Build Coastguard Worker return 0;
410*d57664e9SAndroid Build Coastguard Worker }
411*d57664e9SAndroid Build Coastguard Worker }
412*d57664e9SAndroid Build Coastguard Worker
413*d57664e9SAndroid Build Coastguard Worker }
414