• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1  /*
2   * Copyright (C) 2016 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  
17  #include "adb.h"
18  
19  #include "command.h"
20  #include "print.h"
21  #include "util.h"
22  
23  #include <errno.h>
24  #include <string.h>
25  #include <stdlib.h>
26  #include <unistd.h>
27  #include <sys/types.h>
28  #include <sys/wait.h>
29  #include <limits.h>
30  
31  #include <iostream>
32  #include <istream>
33  #include <streambuf>
34  
35  using namespace std;
36  
37  struct Buffer: public streambuf
38  {
39      Buffer(char* begin, size_t size);
40  };
41  
Buffer(char * begin,size_t size)42  Buffer::Buffer(char* begin, size_t size)
43  {
44      this->setg(begin, begin, begin + size);
45  }
46  
47  int
run_adb(const char * first,...)48  run_adb(const char* first, ...)
49  {
50      Command cmd("adb");
51  
52      if (first == NULL) {
53          return 0;
54      }
55  
56      cmd.AddArg(first);
57  
58      va_list args;
59      va_start(args, first);
60      while (true) {
61          const char* arg = va_arg(args, char*);
62          if (arg == NULL) {
63              break;
64          }
65          cmd.AddArg(arg);
66      }
67      va_end(args);
68  
69      return run_command(cmd);
70  }
71  
72  string
get_system_property(const string & name,int * err)73  get_system_property(const string& name, int* err)
74  {
75      Command cmd("adb");
76      cmd.AddArg("exec-out");
77      cmd.AddArg("getprop");
78      cmd.AddArg(name);
79  
80      return trim(get_command_output(cmd, err, false));
81  }
82  
83  
84  static uint64_t
read_varint(int fd,int * err,bool * done)85  read_varint(int fd, int* err, bool* done)
86  {
87      uint32_t bits = 0;
88      uint64_t result = 0;
89      while (true) {
90          uint8_t byte;
91          ssize_t amt = read(fd, &byte, 1);
92          if (amt == 0) {
93              *done = true;
94              return result;
95          } else if (amt < 0) {
96              return *err = errno;
97          }
98          result |= uint64_t(byte & 0x7F) << bits;
99          if ((byte & 0x80) == 0) {
100              return result;
101          }
102          bits += 7;
103          if (bits > 64) {
104              *err = -1;
105              return 0;
106          }
107      }
108  }
109  
110  static char*
read_sized_buffer(int fd,int * err,size_t * resultSize)111  read_sized_buffer(int fd, int* err, size_t* resultSize)
112  {
113      bool done = false;
114      uint64_t size = read_varint(fd, err, &done);
115      if (*err != 0 || done) {
116          return NULL;
117      }
118      if (size == 0) {
119          *resultSize = 0;
120          return NULL;
121      }
122      // 10 MB seems like a reasonable limit.
123      if (size > 10*1024*1024) {
124          print_error("result buffer too large: %llu", size);
125          return NULL;
126      }
127      char* buf = (char*)malloc(size);
128      if (buf == NULL) {
129          print_error("Can't allocate a buffer of size for test results: %llu", size);
130          return NULL;
131      }
132      int pos = 0;
133      while (size - pos > 0) {
134          ssize_t amt = read(fd, buf+pos, size-pos);
135          if (amt == 0) {
136              // early end of pipe
137              print_error("Early end of pipe.");
138              *err = -1;
139              free(buf);
140              return NULL;
141          } else if (amt < 0) {
142              // error
143              *err = errno;
144              free(buf);
145              return NULL;
146          }
147          pos += amt;
148      }
149      *resultSize = (size_t)size;
150      return buf;
151  }
152  
153  static int
read_sized_proto(int fd,Message * message)154  read_sized_proto(int fd, Message* message)
155  {
156      int err = 0;
157      size_t size;
158      char* buf = read_sized_buffer(fd, &err, &size);
159      if (err != 0) {
160          if (buf != NULL) {
161              free(buf);
162          }
163          return err;
164      } else if (size == 0) {
165          if (buf != NULL) {
166              free(buf);
167          }
168          return 0;
169      } else if (buf == NULL) {
170          return -1;
171      }
172      Buffer buffer(buf, size);
173      istream in(&buffer);
174  
175      err = message->ParseFromIstream(&in) ? 0 : -1;
176  
177      free(buf);
178      return err;
179  }
180  
181  static int
skip_bytes(int fd,ssize_t size,char * scratch,int scratchSize)182  skip_bytes(int fd, ssize_t size, char* scratch, int scratchSize)
183  {
184      while (size > 0) {
185          ssize_t amt = size < scratchSize ? size : scratchSize;
186          fprintf(stderr, "skipping %lu/%ld bytes\n", size, amt);
187          amt = read(fd, scratch, amt);
188          if (amt == 0) {
189              // early end of pipe
190              print_error("Early end of pipe.");
191              return -1;
192          } else if (amt < 0) {
193              // error
194              return errno;
195          }
196          size -= amt;
197      }
198      return 0;
199  }
200  
201  static int
skip_unknown_field(int fd,uint64_t tag,char * scratch,int scratchSize)202  skip_unknown_field(int fd, uint64_t tag, char* scratch, int scratchSize) {
203      bool done = false;
204      int err;
205      uint64_t size;
206      switch (tag & 0x7) {
207          case 0: // varint
208              read_varint(fd, &err, &done);
209              if (err != 0) {
210                  return err;
211              } else if (done) {
212                  return -1;
213              } else {
214                  return 0;
215              }
216          case 1:
217              return skip_bytes(fd, 8, scratch, scratchSize);
218          case 2:
219              size = read_varint(fd, &err, &done);
220              if (err != 0) {
221                  return err;
222              } else if (done) {
223                  return -1;
224              }
225              if (size > INT_MAX) {
226                  // we'll be here a long time but this keeps it from overflowing
227                  return -1;
228              }
229              return skip_bytes(fd, (ssize_t)size, scratch, scratchSize);
230          case 5:
231              return skip_bytes(fd, 4, scratch, scratchSize);
232          default:
233              print_error("bad wire type for tag 0x%lx\n", tag);
234              return -1;
235      }
236  }
237  
238  static int
read_instrumentation_results(int fd,char * scratch,int scratchSize,InstrumentationCallbacks * callbacks)239  read_instrumentation_results(int fd, char* scratch, int scratchSize,
240          InstrumentationCallbacks* callbacks)
241  {
242      bool done = false;
243      int err = 0;
244      string result;
245      while (true) {
246          uint64_t tag = read_varint(fd, &err, &done);
247          if (done) {
248              // Done reading input (this is the only place that a stream end isn't an error).
249              return 0;
250          } else if (err != 0) {
251              return err;
252          } else if (tag == 0xa) { // test_status
253              TestStatus status;
254              err = read_sized_proto(fd, &status);
255              if (err != 0) {
256                  return err;
257              }
258              callbacks->OnTestStatus(status);
259          } else if (tag == 0x12) { // session_status
260              SessionStatus status;
261              err = read_sized_proto(fd, &status);
262              if (err != 0) {
263                  return err;
264              }
265              callbacks->OnSessionStatus(status);
266          } else {
267              err = skip_unknown_field(fd, tag, scratch, scratchSize);
268              if (err != 0) {
269                  return err;
270              }
271          }
272      }
273      return 0;
274  }
275  
276  int
run_instrumentation_test(const string & packageName,const string & runner,const string & className,InstrumentationCallbacks * callbacks)277  run_instrumentation_test(const string& packageName, const string& runner, const string& className,
278          InstrumentationCallbacks* callbacks)
279  {
280      Command cmd("adb");
281      cmd.AddArg("exec-out");
282      cmd.AddArg("am");
283      cmd.AddArg("instrument");
284      cmd.AddArg("-w");
285      cmd.AddArg("-m");
286      const int classLen = className.length();
287      if (classLen > 0) {
288          if (classLen > 1 && className[classLen - 1] == '.') {
289              cmd.AddArg("-e");
290              cmd.AddArg("package");
291  
292              // "am" actually accepts without removing the last ".", but for cleanlines...
293              cmd.AddArg(className.substr(0, classLen - 1));
294          } else {
295              cmd.AddArg("-e");
296              cmd.AddArg("class");
297              cmd.AddArg(className);
298          }
299      }
300      cmd.AddArg(packageName + "/" + runner);
301  
302      print_command(cmd);
303  
304      int fds[2];
305      if (0 != pipe(fds)) {
306          return errno;
307      }
308  
309      pid_t pid = fork();
310  
311      if (pid == -1) {
312          // fork error
313          return errno;
314      } else if (pid == 0) {
315          // child
316          while ((dup2(fds[1], STDOUT_FILENO) == -1) && (errno == EINTR)) {}
317          close(fds[1]);
318          close(fds[0]);
319          const char* prog = cmd.GetProg();
320          char* const* argv = cmd.GetArgv();
321          char* const* env = cmd.GetEnv();
322          exec_with_path_search(prog, argv, env);
323          print_error("Unable to run command: %s", prog);
324          exit(1);
325      } else {
326          // parent
327          close(fds[1]);
328          string result;
329          const int size = 16*1024;
330          char* buf = (char*)malloc(size);
331          int err = read_instrumentation_results(fds[0], buf, size, callbacks);
332          free(buf);
333          int status;
334          waitpid(pid, &status, 0);
335          if (err != 0) {
336              return err;
337          }
338          if (WIFEXITED(status)) {
339              return WEXITSTATUS(status);
340          } else {
341              return -1;
342          }
343      }
344  }
345  
346  /**
347   * Get the second to last bundle in the args list. Stores the last name found
348   * in last. If the path is not found or if the args list is empty, returns NULL.
349   */
350  static const ResultsBundleEntry *
find_penultimate_entry(const ResultsBundle & bundle,va_list args)351  find_penultimate_entry(const ResultsBundle& bundle, va_list args)
352  {
353      const ResultsBundle* b = &bundle;
354      const char* arg = va_arg(args, char*);
355      while (arg) {
356          string last = arg;
357          arg = va_arg(args, char*);
358          bool found = false;
359          for (int i=0; i<b->entries_size(); i++) {
360              const ResultsBundleEntry& e = b->entries(i);
361              if (e.key() == last) {
362                  if (arg == NULL) {
363                      return &e;
364                  } else if (e.has_value_bundle()) {
365                      b = &e.value_bundle();
366                      found = true;
367                  }
368              }
369          }
370          if (!found) {
371              return NULL;
372          }
373          if (arg == NULL) {
374              return NULL;
375          }
376      }
377      return NULL;
378  }
379  
380  string
get_bundle_string(const ResultsBundle & bundle,bool * found,...)381  get_bundle_string(const ResultsBundle& bundle, bool* found, ...)
382  {
383      va_list args;
384      va_start(args, found);
385      const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args);
386      va_end(args);
387      if (entry == NULL) {
388          *found = false;
389          return string();
390      }
391      if (entry->has_value_string()) {
392          *found = true;
393          return entry->value_string();
394      }
395      *found = false;
396      return string();
397  }
398  
399  int32_t
get_bundle_int(const ResultsBundle & bundle,bool * found,...)400  get_bundle_int(const ResultsBundle& bundle, bool* found, ...)
401  {
402      va_list args;
403      va_start(args, found);
404      const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args);
405      va_end(args);
406      if (entry == NULL) {
407          *found = false;
408          return 0;
409      }
410      if (entry->has_value_int()) {
411          *found = true;
412          return entry->value_int();
413      }
414      *found = false;
415      return 0;
416  }
417  
418  float
get_bundle_float(const ResultsBundle & bundle,bool * found,...)419  get_bundle_float(const ResultsBundle& bundle, bool* found, ...)
420  {
421      va_list args;
422      va_start(args, found);
423      const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args);
424      va_end(args);
425      if (entry == NULL) {
426          *found = false;
427          return 0;
428      }
429      if (entry->has_value_float()) {
430          *found = true;
431          return entry->value_float();
432      }
433      *found = false;
434      return 0;
435  }
436  
437  double
get_bundle_double(const ResultsBundle & bundle,bool * found,...)438  get_bundle_double(const ResultsBundle& bundle, bool* found, ...)
439  {
440      va_list args;
441      va_start(args, found);
442      const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args);
443      va_end(args);
444      if (entry == NULL) {
445          *found = false;
446          return 0;
447      }
448      if (entry->has_value_double()) {
449          *found = true;
450          return entry->value_double();
451      }
452      *found = false;
453      return 0;
454  }
455  
456  int64_t
get_bundle_long(const ResultsBundle & bundle,bool * found,...)457  get_bundle_long(const ResultsBundle& bundle, bool* found, ...)
458  {
459      va_list args;
460      va_start(args, found);
461      const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args);
462      va_end(args);
463      if (entry == NULL) {
464          *found = false;
465          return 0;
466      }
467      if (entry->has_value_long()) {
468          *found = true;
469          return entry->value_long();
470      }
471      *found = false;
472      return 0;
473  }
474  
475