/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "adb.h" #include "command.h" #include "print.h" #include "util.h" #include <errno.h> #include <string.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/wait.h> #include <limits.h> #include <iostream> #include <istream> #include <streambuf> using namespace std; struct Buffer: public streambuf { Buffer(char* begin, size_t size); }; Buffer::Buffer(char* begin, size_t size) { this->setg(begin, begin, begin + size); } int run_adb(const char* first, ...) { Command cmd("adb"); if (first == NULL) { return 0; } cmd.AddArg(first); va_list args; va_start(args, first); while (true) { const char* arg = va_arg(args, char*); if (arg == NULL) { break; } cmd.AddArg(arg); } va_end(args); return run_command(cmd); } string get_system_property(const string& name, int* err) { Command cmd("adb"); cmd.AddArg("shell"); cmd.AddArg("getprop"); cmd.AddArg(name); return trim(get_command_output(cmd, err, false)); } static uint64_t read_varint(int fd, int* err, bool* done) { uint32_t bits = 0; uint64_t result = 0; while (true) { uint8_t byte; ssize_t amt = read(fd, &byte, 1); if (amt == 0) { *done = true; return result; } else if (amt < 0) { return *err = errno; } result |= uint64_t(byte & 0x7F) << bits; if ((byte & 0x80) == 0) { return result; } bits += 7; if (bits > 64) { *err = -1; return 0; } } } static char* read_sized_buffer(int fd, int* err, size_t* resultSize) { bool done = false; uint64_t size = read_varint(fd, err, &done); if (*err != 0 || done) { return NULL; } if (size == 0) { *resultSize = 0; return NULL; } // 10 MB seems like a reasonable limit. if (size > 10*1024*1024) { print_error("result buffer too large: %llu", size); return NULL; } char* buf = (char*)malloc(size); if (buf == NULL) { print_error("Can't allocate a buffer of size for test results: %llu", size); return NULL; } int pos = 0; while (size - pos > 0) { ssize_t amt = read(fd, buf+pos, size-pos); if (amt == 0) { // early end of pipe print_error("Early end of pipe."); *err = -1; free(buf); return NULL; } else if (amt < 0) { // error *err = errno; free(buf); return NULL; } pos += amt; } *resultSize = (size_t)size; return buf; } static int read_sized_proto(int fd, Message* message) { int err = 0; size_t size; char* buf = read_sized_buffer(fd, &err, &size); if (err != 0) { if (buf != NULL) { free(buf); } return err; } else if (size == 0) { if (buf != NULL) { free(buf); } return 0; } else if (buf == NULL) { return -1; } Buffer buffer(buf, size); istream in(&buffer); err = message->ParseFromIstream(&in) ? 0 : -1; free(buf); return err; } static int skip_bytes(int fd, ssize_t size, char* scratch, int scratchSize) { while (size > 0) { ssize_t amt = size < scratchSize ? size : scratchSize; fprintf(stderr, "skipping %lu/%ld bytes\n", size, amt); amt = read(fd, scratch, amt); if (amt == 0) { // early end of pipe print_error("Early end of pipe."); return -1; } else if (amt < 0) { // error return errno; } size -= amt; } return 0; } static int skip_unknown_field(int fd, uint64_t tag, char* scratch, int scratchSize) { bool done; int err; uint64_t size; switch (tag & 0x7) { case 0: // varint read_varint(fd, &err, &done); if (err != 0) { return err; } else if (done) { return -1; } else { return 0; } case 1: return skip_bytes(fd, 8, scratch, scratchSize); case 2: size = read_varint(fd, &err, &done); if (err != 0) { return err; } else if (done) { return -1; } if (size > INT_MAX) { // we'll be here a long time but this keeps it from overflowing return -1; } return skip_bytes(fd, (ssize_t)size, scratch, scratchSize); case 5: return skip_bytes(fd, 4, scratch, scratchSize); default: print_error("bad wire type for tag 0x%lx\n", tag); return -1; } } static int read_instrumentation_results(int fd, char* scratch, int scratchSize, InstrumentationCallbacks* callbacks) { bool done = false; int err = 0; string result; while (true) { uint64_t tag = read_varint(fd, &err, &done); if (done) { // Done reading input (this is the only place that a stream end isn't an error). return 0; } else if (err != 0) { return err; } else if (tag == 0xa) { // test_status TestStatus status; err = read_sized_proto(fd, &status); if (err != 0) { return err; } callbacks->OnTestStatus(status); } else if (tag == 0x12) { // session_status SessionStatus status; err = read_sized_proto(fd, &status); if (err != 0) { return err; } callbacks->OnSessionStatus(status); } else { err = skip_unknown_field(fd, tag, scratch, scratchSize); if (err != 0) { return err; } } } return 0; } int run_instrumentation_test(const string& packageName, const string& runner, const string& className, InstrumentationCallbacks* callbacks) { Command cmd("adb"); cmd.AddArg("shell"); cmd.AddArg("am"); cmd.AddArg("instrument"); cmd.AddArg("-w"); cmd.AddArg("-m"); if (className.length() > 0) { cmd.AddArg("-e"); cmd.AddArg("class"); cmd.AddArg(className); } cmd.AddArg(packageName + "/" + runner); print_command(cmd); int fds[2]; pipe(fds); pid_t pid = fork(); if (pid == -1) { // fork error return errno; } else if (pid == 0) { // child while ((dup2(fds[1], STDOUT_FILENO) == -1) && (errno == EINTR)) {} close(fds[1]); close(fds[0]); const char* prog = cmd.GetProg(); char* const* argv = cmd.GetArgv(); char* const* env = cmd.GetEnv(); exec_with_path_search(prog, argv, env); print_error("Unable to run command: %s", prog); exit(1); } else { // parent close(fds[1]); string result; const int size = 16*1024; char* buf = (char*)malloc(size); int err = read_instrumentation_results(fds[0], buf, size, callbacks); free(buf); int status; waitpid(pid, &status, 0); if (err != 0) { return err; } if (WIFEXITED(status)) { return WEXITSTATUS(status); } else { return -1; } } } /** * Get the second to last bundle in the args list. Stores the last name found * in last. If the path is not found or if the args list is empty, returns NULL. */ static const ResultsBundleEntry * find_penultimate_entry(const ResultsBundle& bundle, va_list args) { const ResultsBundle* b = &bundle; const char* arg = va_arg(args, char*); while (arg) { string last = arg; arg = va_arg(args, char*); bool found = false; for (int i=0; i<b->entries_size(); i++) { const ResultsBundleEntry& e = b->entries(i); if (e.key() == last) { if (arg == NULL) { return &e; } else if (e.has_value_bundle()) { b = &e.value_bundle(); found = true; } } } if (!found) { return NULL; } if (arg == NULL) { return NULL; } } return NULL; } string get_bundle_string(const ResultsBundle& bundle, bool* found, ...) { va_list args; va_start(args, found); const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args); va_end(args); if (entry == NULL) { *found = false; return string(); } if (entry->has_value_string()) { *found = true; return entry->value_string(); } *found = false; return string(); } int32_t get_bundle_int(const ResultsBundle& bundle, bool* found, ...) { va_list args; va_start(args, found); const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args); va_end(args); if (entry == NULL) { *found = false; return 0; } if (entry->has_value_int()) { *found = true; return entry->value_int(); } *found = false; return 0; } float get_bundle_float(const ResultsBundle& bundle, bool* found, ...) { va_list args; va_start(args, found); const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args); va_end(args); if (entry == NULL) { *found = false; return 0; } if (entry->has_value_float()) { *found = true; return entry->value_float(); } *found = false; return 0; } double get_bundle_double(const ResultsBundle& bundle, bool* found, ...) { va_list args; va_start(args, found); const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args); va_end(args); if (entry == NULL) { *found = false; return 0; } if (entry->has_value_double()) { *found = true; return entry->value_double(); } *found = false; return 0; } int64_t get_bundle_long(const ResultsBundle& bundle, bool* found, ...) { va_list args; va_start(args, found); const ResultsBundleEntry* entry = find_penultimate_entry(bundle, args); va_end(args); if (entry == NULL) { *found = false; return 0; } if (entry->has_value_long()) { *found = true; return entry->value_long(); } *found = false; return 0; }