xref: /aosp_15_r20/frameworks/base/cmds/incident/main.cpp (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
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