diff options
Diffstat (limited to 'adb')
| -rw-r--r-- | adb/Android.mk | 4 | ||||
| -rw-r--r-- | adb/SERVICES.TXT | 27 | ||||
| -rw-r--r-- | adb/adb.c | 227 | ||||
| -rw-r--r-- | adb/adb_client.c | 19 | ||||
| -rw-r--r-- | adb/adb_client.h | 4 | ||||
| -rw-r--r-- | adb/commandline.c | 139 | ||||
| -rw-r--r-- | adb/sysdeps_win32.c | 2 | 
7 files changed, 369 insertions, 53 deletions
| diff --git a/adb/Android.mk b/adb/Android.mk index bc8315e..32dd95a 100644 --- a/adb/Android.mk +++ b/adb/Android.mk @@ -16,7 +16,7 @@ EXTRA_SRCS :=  ifeq ($(HOST_OS),linux)    USB_SRCS := usb_linux.c    EXTRA_SRCS := get_my_path_linux.c -  LOCAL_LDLIBS += -lrt -lncurses -lpthread +  LOCAL_LDLIBS += -lrt -ldl -lpthread  endif  ifeq ($(HOST_OS),darwin) @@ -139,7 +139,7 @@ include $(BUILD_EXECUTABLE)  ifneq ($(SDK_ONLY),true)  include $(CLEAR_VARS) -LOCAL_LDLIBS := -lrt -lncurses -lpthread +LOCAL_LDLIBS := -lrt -ldl -lpthread  LOCAL_SRC_FILES := \  	adb.c \ diff --git a/adb/SERVICES.TXT b/adb/SERVICES.TXT index d9aa09c..b53bc44 100644 --- a/adb/SERVICES.TXT +++ b/adb/SERVICES.TXT @@ -117,7 +117,34 @@ host:<request>      or even any one of the local services described below. +<host-prefix>:forward:norebind:<local>;<remote> +    Same as <host-prefix>:forward:<local>;<remote> except that it will +    fail it there is already a forward connection from <local>. +    Used to implement 'adb forward --no-rebind <local> <remote>' + +<host-prefix>:killforward:<local> +    Remove any existing forward local connection from <local>. +    This is used to implement 'adb forward --remove <local>' + +<host-prefix>:killforward-all +    Remove all forward network connections. +    This is used to implement 'adb forward --remove-all'. + +<host-prefix>:list-forward +    List all existing forward connections from this server. +    This returns something that looks like the following: + +       <hex4>: The length of the payload, as 4 hexadecimal chars. +       <payload>: A series of lines of the following format: + +         <serial> " " <local> " " <remote> "\n" + +    Where <serial> is a device serial number. +          <local>  is the host-specific endpoint (e.g. tcp:9000). +          <remote> is the device-specific endpoint. + +    Used to implement 'adb forward --list'.  LOCAL SERVICES: @@ -25,6 +25,7 @@  #include <string.h>  #include <time.h>  #include <sys/time.h> +#include <stdint.h>  #include "sysdeps.h"  #include "adb.h" @@ -34,7 +35,7 @@  #if !ADB_HOST  #include <private/android_filesystem_config.h> -#include <linux/capability.h> +#include <sys/capability.h>  #include <linux/prctl.h>  #include <sys/mount.h>  #else @@ -46,6 +47,7 @@ ADB_MUTEX_DEFINE( D_lock );  #endif  int HOST = 0; +int gListenAll = 0;  static int auth_enabled = 0; @@ -701,7 +703,13 @@ int local_name_to_fd(const char *name)      if(!strncmp("tcp:", name, 4)){          int  ret;          port = atoi(name + 4); -        ret = socket_loopback_server(port, SOCK_STREAM); + +        if (gListenAll > 0) { +            ret = socket_inaddr_any_server(port, SOCK_STREAM); +        } else { +            ret = socket_loopback_server(port, SOCK_STREAM); +        } +          return ret;      }  #ifndef HAVE_WIN32_IPC  /* no Unix-domain sockets on Win32 */ @@ -722,24 +730,90 @@ int local_name_to_fd(const char *name)      return -1;  } -static int remove_listener(const char *local_name, const char *connect_to, atransport* transport) +// Write a single line describing a listener to a user-provided buffer. +// Appends a trailing zero, even in case of truncation, but the function +// returns the full line length. +// If |buffer| is NULL, does not write but returns required size. +static int format_listener(alistener* l, char* buffer, size_t buffer_len) { +    // Format is simply: +    // +    //  <device-serial> " " <local-name> " " <remote-name> "\n" +    // +    int local_len = strlen(l->local_name); +    int connect_len = strlen(l->connect_to); +    int serial_len = strlen(l->transport->serial); + +    if (buffer != NULL) { +        snprintf(buffer, buffer_len, "%s %s %s\n", +                l->transport->serial, l->local_name, l->connect_to); +    } +    // NOTE: snprintf() on Windows returns -1 in case of truncation, so +    // return the computed line length instead. +    return local_len + connect_len + serial_len + 3; +} + +// Write the list of current listeners (network redirections) into a +// user-provided buffer. Appends a trailing zero, even in case of +// trunctaion, but return the full size in bytes. +// If |buffer| is NULL, does not write but returns required size. +static int format_listeners(char* buf, size_t buflen) +{ +    alistener* l; +    int result = 0; +    for (l = listener_list.next; l != &listener_list; l = l->next) { +        // Ignore special listeners like those for *smartsocket* +        if (l->connect_to[0] == '*') +          continue; +        int len = format_listener(l, buf, buflen); +        // Ensure there is space for the trailing zero. +        result += len; +        if (buf != NULL) { +          buf += len; +          buflen -= len; +          if (buflen <= 0) +              break; +        } +    } +    return result; +} + +static int remove_listener(const char *local_name, atransport* transport)  {      alistener *l;      for (l = listener_list.next; l != &listener_list; l = l->next) { -        if (!strcmp(local_name, l->local_name) && -            !strcmp(connect_to, l->connect_to) && -            l->transport && l->transport == transport) { - -            listener_disconnect(l, transport); +        if (!strcmp(local_name, l->local_name)) { +            listener_disconnect(l, l->transport);              return 0;          }      } -      return -1;  } -static int install_listener(const char *local_name, const char *connect_to, atransport* transport) +static void remove_all_listeners(void) +{ +    alistener *l, *l_next; +    for (l = listener_list.next; l != &listener_list; l = l_next) { +        l_next = l->next; +        // Never remove smart sockets. +        if (l->connect_to[0] == '*') +            continue; +        listener_disconnect(l, l->transport); +    } +} + +// error/status codes for install_listener. +typedef enum { +  INSTALL_STATUS_OK = 0, +  INSTALL_STATUS_INTERNAL_ERROR = -1, +  INSTALL_STATUS_CANNOT_BIND = -2, +  INSTALL_STATUS_CANNOT_REBIND = -3, +} install_status_t; + +static install_status_t install_listener(const char *local_name, +                                         const char *connect_to, +                                         atransport* transport, +                                         int no_rebind)  {      alistener *l; @@ -751,12 +825,17 @@ static int install_listener(const char *local_name, const char *connect_to, atra                  /* can't repurpose a smartsocket */              if(l->connect_to[0] == '*') { -                return -1; +                return INSTALL_STATUS_INTERNAL_ERROR; +            } + +                /* can't repurpose a listener if 'no_rebind' is true */ +            if (no_rebind) { +                return INSTALL_STATUS_CANNOT_REBIND;              }              cto = strdup(connect_to);              if(cto == 0) { -                return -1; +                return INSTALL_STATUS_INTERNAL_ERROR;              }              //printf("rebinding '%s' to '%s'\n", local_name, connect_to); @@ -767,7 +846,7 @@ static int install_listener(const char *local_name, const char *connect_to, atra                  l->transport = transport;                  add_transport_disconnect(l->transport, &l->disconnect);              } -            return 0; +            return INSTALL_STATUS_OK;          }      } @@ -804,11 +883,11 @@ static int install_listener(const char *local_name, const char *connect_to, atra          l->disconnect.func   = listener_disconnect;          add_transport_disconnect(transport, &l->disconnect);      } -    return 0; +    return INSTALL_STATUS_OK;  nomem:      fatal("cannot allocate listener"); -    return 0; +    return INSTALL_STATUS_INTERNAL_ERROR;  }  #ifdef HAVE_WIN32_PROC @@ -913,6 +992,7 @@ int launch_server(int server_port)      /* message since the pipe handles must be inheritable, we use a     */      /* security attribute                                               */      HANDLE                pipe_read, pipe_write; +    HANDLE                stdout_handle, stderr_handle;      SECURITY_ATTRIBUTES   sa;      STARTUPINFO           startup;      PROCESS_INFORMATION   pinfo; @@ -932,6 +1012,26 @@ int launch_server(int server_port)      SetHandleInformation( pipe_read, HANDLE_FLAG_INHERIT, 0 ); +    /* Some programs want to launch an adb command and collect its output by +     * calling CreateProcess with inheritable stdout/stderr handles, then +     * using read() to get its output. When this happens, the stdout/stderr +     * handles passed to the adb client process will also be inheritable. +     * When starting the adb server here, care must be taken to reset them +     * to non-inheritable. +     * Otherwise, something bad happens: even if the adb command completes, +     * the calling process is stuck while read()-ing from the stdout/stderr +     * descriptors, because they're connected to corresponding handles in the +     * adb server process (even if the latter never uses/writes to them). +     */ +    stdout_handle = GetStdHandle( STD_OUTPUT_HANDLE ); +    stderr_handle = GetStdHandle( STD_ERROR_HANDLE ); +    if (stdout_handle != INVALID_HANDLE_VALUE) { +        SetHandleInformation( stdout_handle, HANDLE_FLAG_INHERIT, 0 ); +    } +    if (stderr_handle != INVALID_HANDLE_VALUE) { +        SetHandleInformation( stderr_handle, HANDLE_FLAG_INHERIT, 0 ); +    } +      ZeroMemory( &startup, sizeof(startup) );      startup.cb = sizeof(startup);      startup.hStdInput  = GetStdHandle( STD_INPUT_HANDLE ); @@ -1008,8 +1108,10 @@ int launch_server(int server_port)          dup2(fd[1], STDERR_FILENO);          adb_close(fd[1]); +        char str_port[30]; +        snprintf(str_port, sizeof(str_port), "%d",  server_port);          // child process -        int result = execl(path, "adb", "fork-server", "server", NULL); +        int result = execl(path, "adb", "-P", str_port, "fork-server", "server", NULL);          // this should not return          fprintf(stderr, "OOPS! execl returned %d, errno: %d\n", result, errno);      } else  { @@ -1113,7 +1215,7 @@ int adb_main(int is_daemon, int server_port)      char local_name[30];      build_local_name(local_name, sizeof(local_name), server_port); -    if(install_listener(local_name, "*smartsocket*", NULL)) { +    if(install_listener(local_name, "*smartsocket*", NULL, 0)) {          exit(1);      }  #else @@ -1136,7 +1238,7 @@ int adb_main(int is_daemon, int server_port)      /* don't run as root if we are running in secure mode */      if (should_drop_privileges()) {          struct __user_cap_header_struct header; -        struct __user_cap_data_struct cap; +        struct __user_cap_data_struct cap[2];          if (prctl(PR_SET_KEEPCAPS, 1, 0, 0, 0) != 0) {              exit(1); @@ -1169,18 +1271,21 @@ int adb_main(int is_daemon, int server_port)              exit(1);          } +        memset(&header, 0, sizeof(header)); +        memset(cap, 0, sizeof(cap)); +          /* set CAP_SYS_BOOT capability, so "adb reboot" will succeed */ -        header.version = _LINUX_CAPABILITY_VERSION; +        header.version = _LINUX_CAPABILITY_VERSION_3;          header.pid = 0; -        cap.effective = cap.permitted = (1 << CAP_SYS_BOOT); -        cap.inheritable = 0; -        capset(&header, &cap); +        cap[CAP_TO_INDEX(CAP_SYS_BOOT)].effective |= CAP_TO_MASK(CAP_SYS_BOOT); +        cap[CAP_TO_INDEX(CAP_SYS_BOOT)].permitted |= CAP_TO_MASK(CAP_SYS_BOOT); +        capset(&header, cap);          D("Local port disabled\n");      } else {          char local_name[30];          build_local_name(local_name, sizeof(local_name), server_port); -        if(install_listener(local_name, "*smartsocket*", NULL)) { +        if(install_listener(local_name, "*smartsocket*", NULL, 0)) {              exit(1);          }      } @@ -1474,24 +1579,63 @@ int handle_host_request(char *service, transport_type ttype, char* serial, int r      }  #endif // ADB_HOST -    if(!strncmp(service,"forward:",8) || !strncmp(service,"killforward:",12)) { +    if(!strcmp(service,"list-forward")) { +        // Create the list of forward redirections. +        char header[9]; +        int buffer_size = format_listeners(NULL, 0); +        // Add one byte for the trailing zero. +        char* buffer = malloc(buffer_size+1); +        (void) format_listeners(buffer, buffer_size+1); +        snprintf(header, sizeof header, "OKAY%04x", buffer_size); +        writex(reply_fd, header, 8); +        writex(reply_fd, buffer, buffer_size); +        free(buffer); +        return 0; +    } + +    if (!strcmp(service,"killforward-all")) { +        remove_all_listeners(); +        adb_write(reply_fd, "OKAYOKAY", 8); +        return 0; +    } + +    if(!strncmp(service,"forward:",8) || +       !strncmp(service,"killforward:",12)) {          char *local, *remote, *err;          int r;          atransport *transport;          int createForward = strncmp(service,"kill",4); +        int no_rebind = 0; -        local = service + (createForward ? 8 : 12); -        remote = strchr(local,';'); -        if(remote == 0) { -            sendfailmsg(reply_fd, "malformed forward spec"); -            return 0; +        local = strchr(service, ':') + 1; + +        // Handle forward:norebind:<local>... here +        if (createForward && !strncmp(local, "norebind:", 9)) { +            no_rebind = 1; +            local = strchr(local, ':') + 1;          } -        *remote++ = 0; -        if((local[0] == 0) || (remote[0] == 0) || (remote[0] == '*')){ -            sendfailmsg(reply_fd, "malformed forward spec"); -            return 0; +        remote = strchr(local,';'); + +        if (createForward) { +            // Check forward: parameter format: '<local>;<remote>' +            if(remote == 0) { +                sendfailmsg(reply_fd, "malformed forward spec"); +                return 0; +            } + +            *remote++ = 0; +            if((local[0] == 0) || (remote[0] == 0) || (remote[0] == '*')){ +                sendfailmsg(reply_fd, "malformed forward spec"); +                return 0; +            } +        } else { +            // Check killforward: parameter format: '<local>' +            if (local[0] == 0) { +                sendfailmsg(reply_fd, "malformed forward spec"); +                return 0; +            }          }          transport = acquire_one_transport(CS_ANY, ttype, serial, &err); @@ -1501,9 +1645,9 @@ int handle_host_request(char *service, transport_type ttype, char* serial, int r          }          if (createForward) { -            r = install_listener(local, remote, transport); +            r = install_listener(local, remote, transport, no_rebind);          } else { -            r = remove_listener(local, remote, transport); +            r = remove_listener(local, transport);          }          if(r == 0) {                  /* 1st OKAY is connect, 2nd OKAY is status */ @@ -1512,7 +1656,18 @@ int handle_host_request(char *service, transport_type ttype, char* serial, int r          }          if (createForward) { -            sendfailmsg(reply_fd, (r == -1) ? "cannot rebind smartsocket" : "cannot bind socket"); +            const char* message; +            switch (r) { +              case INSTALL_STATUS_CANNOT_BIND: +                message = "cannot bind to socket"; +                break; +              case INSTALL_STATUS_CANNOT_REBIND: +                message = "cannot rebind existing socket"; +                break; +              default: +                message = "internal error"; +            } +            sendfailmsg(reply_fd, message);          } else {              sendfailmsg(reply_fd, "cannot remove listener");          } diff --git a/adb/adb_client.c b/adb/adb_client.c index 9a812f0..8340738 100644 --- a/adb/adb_client.c +++ b/adb/adb_client.c @@ -17,6 +17,7 @@ static transport_type __adb_transport = kTransportAny;  static const char* __adb_serial = NULL;  static int __adb_server_port = DEFAULT_ADB_PORT; +static const char* __adb_server_name = NULL;  void adb_set_transport(transport_type type, const char* serial)  { @@ -29,6 +30,11 @@ void adb_set_tcp_specifics(int server_port)      __adb_server_port = server_port;  } +void adb_set_tcp_name(const char* hostname) +{ +    __adb_server_name = hostname; +} +  int  adb_get_emulator_console_port(void)  {      const char*   serial = __adb_serial; @@ -181,7 +187,11 @@ int _adb_connect(const char *service)      }      snprintf(tmp, sizeof tmp, "%04x", len); -    fd = socket_loopback_client(__adb_server_port, SOCK_STREAM); +    if (__adb_server_name) +        fd = socket_network_client(__adb_server_name, __adb_server_port, SOCK_STREAM); +    else +        fd = socket_loopback_client(__adb_server_port, SOCK_STREAM); +      if(fd < 0) {          strcpy(__adb_error, "cannot connect to daemon");          return -2; @@ -212,7 +222,10 @@ int adb_connect(const char *service)      int fd = _adb_connect("host:version");      D("adb_connect: service %s\n", service); -    if(fd == -2) { +    if(fd == -2 && __adb_server_name) { +        fprintf(stderr,"** Cannot start server on remote host\n"); +        return fd; +    } else if(fd == -2) {          fprintf(stdout,"* daemon not running. starting it now on port %d *\n",                  __adb_server_port);      start_server: @@ -266,7 +279,7 @@ int adb_connect(const char *service)      fd = _adb_connect(service);      if(fd == -2) { -        fprintf(stderr,"** daemon still not running"); +        fprintf(stderr,"** daemon still not running\n");      }      D("adb_connect: return fd %d\n", fd); diff --git a/adb/adb_client.h b/adb/adb_client.h index 40ab189..0ec47ca 100644 --- a/adb/adb_client.h +++ b/adb/adb_client.h @@ -29,6 +29,10 @@ void adb_set_transport(transport_type type, const char* serial);  */  void adb_set_tcp_specifics(int server_port); +/* Set TCP Hostname of the transport to use +*/ +void adb_set_tcp_name(const char* hostname); +  /* Return the console port of the currently connected emulator (if any)   * of -1 if there is no emulator, and -2 if there is more than one.   * assumes adb_set_transport() was alled previously... diff --git a/adb/commandline.c b/adb/commandline.c index 24cbb5a..a927423 100644 --- a/adb/commandline.c +++ b/adb/commandline.c @@ -46,6 +46,7 @@ int install_app(transport_type transport, char* serial, int argc, char** argv);  int uninstall_app(transport_type transport, char* serial, int argc, char** argv);  static const char *gProductOutPath = NULL; +extern int gListenAll;  static char *product_file(const char *extra)  { @@ -80,6 +81,7 @@ void help()      fprintf(stderr,          "\n" +        " -a                            - directs adb to listen on all interfaces for a connection\n"          " -d                            - directs command to the only connected USB device\n"          "                                 returns an error if more than one USB device is present.\n"          " -e                            - directs command to the only running emulator.\n" @@ -93,6 +95,8 @@ void help()          "                                 If -p is not specified, the ANDROID_PRODUCT_OUT\n"          "                                 environment variable is used, which must\n"          "                                 be an absolute path.\n" +        " -H                            - Name of adb server host (default: localhost)\n" +        " -P                            - Port of adb server (default: 5037)\n"          " devices [-l]                  - list all connected devices\n"          "                                 ('-l' will also list device qualifiers)\n"          " connect <host>[:<port>]       - connect to a device via TCP/IP\n" @@ -112,6 +116,9 @@ void help()          "  adb shell <command>          - run remote shell command\n"          "  adb emu <command>            - run emulator console command\n"          "  adb logcat [ <filter-spec> ] - View device log\n" +        "  adb forward --list           - list all forward socket connections.\n" +        "                                 the format is a list of lines with the following format:\n" +        "                                    <serial> \" \" <local> \" \" <remote> \"\\n\"\n"          "  adb forward <local> <remote> - forward socket connections\n"          "                                 forward specs are one of: \n"          "                                   tcp:<port>\n" @@ -120,6 +127,11 @@ void help()          "                                   localfilesystem:<unix domain socket name>\n"          "                                   dev:<character device name>\n"          "                                   jdwp:<process pid> (remote only)\n" +        "  adb forward --no-rebind <local> <remote>\n" +        "                               - same as 'adb forward <local> <remote>' but fails\n" +        "                                 if <local> is already forwarded\n" +        "  adb forward --remove <local> - remove a specific forward socket connection\n" +        "  adb forward --remove-all     - remove all forward socket connections\n"          "  adb jdwp                     - list PIDs of processes hosting a JDWP transport\n"          "  adb install [-l] [-r] [-s] [--algo <algorithm name> --key <hex-encoded key> --iv <hex-encoded iv>] <file>\n"          "                               - push this package file to the device and install it\n" @@ -371,7 +383,7 @@ static void format_host_command(char* buffer, size_t  buflen, const char* comman      }  } -int adb_download_buffer(const char *service, const void* data, int sz, +int adb_download_buffer(const char *service, const char *fn, const void* data, int sz,                          unsigned progress)  {      char buf[4096]; @@ -407,7 +419,7 @@ int adb_download_buffer(const char *service, const void* data, int sz,          sz -= xfer;          ptr += xfer;          if(progress) { -            printf("sending: '%s' %4d%%    \r", service, (int)(100LL - ((100LL * sz) / (total)))); +            printf("sending: '%s' %4d%%    \r", fn, (int)(100LL - ((100LL * sz) / (total))));              fflush(stdout);          }      } @@ -439,11 +451,11 @@ int adb_download(const char *service, const char *fn, unsigned progress)      data = load_file(fn, &sz);      if(data == 0) { -        fprintf(stderr,"* cannot read '%s' *\n", service); +        fprintf(stderr,"* cannot read '%s' *\n", fn);          return -1;      } -    int status = adb_download_buffer(service, data, sz, progress); +    int status = adb_download_buffer(service, fn, data, sz, progress);      free(data);      return status;  } @@ -938,9 +950,9 @@ int adb_commandline(int argc, char **argv)      int server_port = DEFAULT_ADB_PORT;      if (server_port_str && strlen(server_port_str) > 0) {          server_port = (int) strtol(server_port_str, NULL, 0); -        if (server_port <= 0) { +        if (server_port <= 0 || server_port > 65535) {              fprintf(stderr, -                    "adb: Env var ANDROID_ADB_SERVER_PORT must be a positive number. Got \"%s\"\n", +                    "adb: Env var ANDROID_ADB_SERVER_PORT must be a positive number less than 65535. Got \"%s\"\n",                      server_port_str);              return usage();          } @@ -986,6 +998,42 @@ int adb_commandline(int argc, char **argv)              ttype = kTransportUsb;          } else if (!strcmp(argv[0],"-e")) {              ttype = kTransportLocal; +        } else if (!strcmp(argv[0],"-a")) { +            gListenAll = 1; +        } else if(!strncmp(argv[0], "-H", 2)) { +            const char *hostname = NULL; +            if (argv[0][2] == '\0') { +                if (argc < 2) return usage(); +                hostname = argv[1]; +                argc--; +                argv++; +            } else { +                hostname = argv[0] + 2; +            } +            adb_set_tcp_name(hostname); + +        } else if(!strncmp(argv[0], "-P", 2)) { +            if (argv[0][2] == '\0') { +                if (argc < 2) return usage(); +                server_port_str = argv[1]; +                argc--; +                argv++; +            } else { +                server_port_str = argv[0] + 2; +            } +            if (strlen(server_port_str) > 0) { +                server_port = (int) strtol(server_port_str, NULL, 0); +                if (server_port <= 0 || server_port > 65535) { +                    fprintf(stderr, +                            "adb: port number must be a positive number less than 65536. Got \"%s\"\n", +                            server_port_str); +                    return usage(); +                } +            } else { +                fprintf(stderr, +                "adb: port number must be a positive number less than 65536. Got empty string.\n"); +                return usage(); +            }          } else {                  /* out of recognized modifiers and flags */              break; @@ -1223,16 +1271,85 @@ top:      }      if(!strcmp(argv[0], "forward")) { -        if(argc != 3) return usage(); +        char host_prefix[64]; +        char remove = 0; +        char remove_all = 0; +        char list = 0; +        char no_rebind = 0; + +        // Parse options here. +        while (argc > 1 && argv[1][0] == '-') { +            if (!strcmp(argv[1], "--list")) +                list = 1; +            else if (!strcmp(argv[1], "--remove")) +                remove = 1; +            else if (!strcmp(argv[1], "--remove-all")) +                remove_all = 1; +            else if (!strcmp(argv[1], "--no-rebind")) +                no_rebind = 1; +            else { +                return usage(); +            } +            argc--; +            argv++; +        } + +        // Ensure we can only use one option at a time. +        if (list + remove + remove_all + no_rebind > 1) { +            return usage(); +        } + +        // Determine the <host-prefix> for this command.          if (serial) { -            snprintf(buf, sizeof buf, "host-serial:%s:forward:%s;%s",serial, argv[1], argv[2]); +            snprintf(host_prefix, sizeof host_prefix, "host-serial:%s", +                    serial);          } else if (ttype == kTransportUsb) { -            snprintf(buf, sizeof buf, "host-usb:forward:%s;%s", argv[1], argv[2]); +            snprintf(host_prefix, sizeof host_prefix, "host-usb");          } else if (ttype == kTransportLocal) { -            snprintf(buf, sizeof buf, "host-local:forward:%s;%s", argv[1], argv[2]); +            snprintf(host_prefix, sizeof host_prefix, "host-local");          } else { -            snprintf(buf, sizeof buf, "host:forward:%s;%s", argv[1], argv[2]); +            snprintf(host_prefix, sizeof host_prefix, "host"); +        } + +        // Implement forward --list +        if (list) { +            if (argc != 1) +                return usage(); +            snprintf(buf, sizeof buf, "%s:list-forward", host_prefix); +            char* forwards = adb_query(buf); +            if (forwards == NULL) { +                fprintf(stderr, "error: %s\n", adb_error()); +                return 1; +            } +            printf("%s", forwards); +            free(forwards); +            return 0;          } + +        // Implement forward --remove-all +        else if (remove_all) { +            if (argc != 1) +                return usage(); +            snprintf(buf, sizeof buf, "%s:killforward-all", host_prefix); +        } + +        // Implement forward --remove <local> +        else if (remove) { +            if (argc != 2) +                return usage(); +            snprintf(buf, sizeof buf, "%s:killforward:%s", host_prefix, argv[1]); +        } +        // Or implement one of: +        //    forward <local> <remote> +        //    forward --no-rebind <local> <remote> +        else +        { +          if (argc != 3) +            return usage(); +          const char* command = no_rebind ? "forward:norebind:" : "forward"; +          snprintf(buf, sizeof buf, "%s:%s:%s;%s", host_prefix, command, argv[1], argv[2]); +        } +          if(adb_command(buf)) {              fprintf(stderr,"error: %s\n", adb_error());              return 1; diff --git a/adb/sysdeps_win32.c b/adb/sysdeps_win32.c index d41c42c..2105b16 100644 --- a/adb/sysdeps_win32.c +++ b/adb/sysdeps_win32.c @@ -781,7 +781,7 @@ int  adb_socket_accept(int  serverfd, struct sockaddr*  addr, socklen_t  *addrle  void  disable_tcp_nagle(int fd)  {      FH   fh = _fh_from_int(fd); -    int  on; +    int  on = 1;      if ( !fh || fh->clazz != &_fh_socket_class )          return; | 
