From 6d2d803003c231df9df3343eb73edf97d9ac3c76 Mon Sep 17 00:00:00 2001 From: yuchenlin Date: Sat, 1 Sep 2018 09:18:06 +0800 Subject: [PATCH 1/2] Notify adb missing There are many user who encounters missing adb. To stop things happens again, we check it and show sexy response to user. Signed-off-by: yuchenlin --- app/src/command.c | 31 +++++++++++++++++++- app/src/command.h | 2 +- app/src/sys/unix/command.c | 58 ++++++++++++++++++++++++++++++++------ app/src/sys/win/command.c | 11 +++++--- 4 files changed, 88 insertions(+), 14 deletions(-) diff --git a/app/src/command.c b/app/src/command.c index b7ea67d23c..011a2b85b1 100644 --- a/app/src/command.c +++ b/app/src/command.c @@ -1,5 +1,8 @@ #include "command.h" +#ifndef __WINDOWS__ +# include +#endif #include #include #include @@ -18,9 +21,30 @@ static inline const char *get_adb_command() { return adb_command; } +static void show_err_msg(int err) { +#ifdef __WINDOWS__ + (void) err; // unused + LOGE("Failed to execute adb"); +#else + switch (err) { + case -1: + LOGE("Failed to execute adb"); + break; + case ENOENT: + LOGE("'adb' command not found (make it accessible from your PATH " + "or define its full path in the ADB environment variable)"); + break; + default: + LOGE("Failed to execute adb: %s", strerror(err)); + break; + } +#endif +} + process_t adb_execute(const char *serial, const char *const adb_cmd[], int len) { const char *cmd[len + 4]; int i; + process_t process; cmd[0] = get_adb_command(); if (serial) { cmd[1] = "-s"; @@ -32,7 +56,12 @@ process_t adb_execute(const char *serial, const char *const adb_cmd[], int len) memcpy(&cmd[i], adb_cmd, len * sizeof(const char *)); cmd[len + i] = NULL; - return cmd_execute(cmd[0], cmd); + int r = cmd_execute(cmd[0], cmd, &process); + if (r != 0) { + show_err_msg(r); + return PROCESS_NONE; + } + return process; } process_t adb_forward(const char *serial, uint16_t local_port, const char *device_socket_name) { diff --git a/app/src/command.h b/app/src/command.h index 4113f2512b..dfc0d58a33 100644 --- a/app/src/command.h +++ b/app/src/command.h @@ -32,7 +32,7 @@ #endif # define NO_EXIT_CODE -1 -process_t cmd_execute(const char *path, const char *const argv[]); +int cmd_execute(const char *path, const char *const argv[], process_t *process); SDL_bool cmd_terminate(process_t pid); SDL_bool cmd_simple_wait(process_t pid, exit_code_t *exit_code); diff --git a/app/src/sys/unix/command.c b/app/src/sys/unix/command.c index 0083a93b3a..20218cb9e3 100644 --- a/app/src/sys/unix/command.c +++ b/app/src/sys/unix/command.c @@ -1,5 +1,7 @@ #include "command.h" +#include +#include #include #include #include @@ -7,18 +9,58 @@ #include #include "log.h" -pid_t cmd_execute(const char *path, const char *const argv[]) { - pid_t pid = fork(); - if (pid == -1) { - perror("fork"); +int cmd_execute(const char *path, const char *const argv[], pid_t *pid) { + int fd[2]; + int ret = 0; + + if (pipe(fd) == -1) { + perror("pipe"); return -1; } - if (pid == 0) { - execvp(path, (char *const *)argv); - perror("exec"); + + *pid = fork(); + if (*pid == -1) { + perror("fork"); + ret = -1; + goto end; + } + + if (*pid > 0) { + // parent close write side + close(fd[1]); + fd[1] = -1; + // wait for EOF or receive errno from child + if (read(fd[0], &ret, sizeof(int)) == -1) { + perror("read"); + ret = -1; + goto end; + } + } else if (*pid == 0) { + // child close read side + close(fd[0]); + if (fcntl(fd[1], F_SETFD, FD_CLOEXEC) == 0) { + execvp(path, (char *const *)argv); + } else { + perror("fcntl"); + } + // send errno to the parent + ret = errno; + if (write(fd[1], &ret, sizeof(int)) == -1) { + perror("write"); + } + // close write side before exiting + close(fd[1]); _exit(1); } - return pid; + +end: + if (fd[0] != -1) { + close(fd[0]); + } + if (fd[1] != -1) { + close(fd[1]); + } + return ret; } SDL_bool cmd_terminate(pid_t pid) { diff --git a/app/src/sys/win/command.c b/app/src/sys/win/command.c index 2552eeca4f..cea928b40d 100644 --- a/app/src/sys/win/command.c +++ b/app/src/sys/win/command.c @@ -4,7 +4,7 @@ #include "log.h" #include "str_util.h" -HANDLE cmd_execute(const char *path, const char *const argv[]) { +int cmd_execute(const char *path, const char *const argv[], HANDLE *handle) { STARTUPINFO si; PROCESS_INFORMATION pi; memset(&si, 0, sizeof(si)); @@ -18,7 +18,8 @@ HANDLE cmd_execute(const char *path, const char *const argv[]) { size_t ret = xstrjoin(cmd, argv, ' ', sizeof(cmd)); if (ret >= sizeof(cmd)) { LOGE("Command too long (%" PRIsizet " chars)", sizeof(cmd) - 1); - return NULL; + *handle = NULL; + return -1; } #ifdef WINDOWS_NOCONSOLE @@ -27,10 +28,12 @@ HANDLE cmd_execute(const char *path, const char *const argv[]) { int flags = 0; #endif if (!CreateProcess(NULL, cmd, NULL, NULL, FALSE, flags, NULL, NULL, &si, &pi)) { - return NULL; + *handle = NULL; + return -1; } - return pi.hProcess; + *handle = pi.hProcess; + return 0; } SDL_bool cmd_terminate(HANDLE handle) { From 55d33ddd5fb0b559eed33d8f47d9b07a2456979a Mon Sep 17 00:00:00 2001 From: Romain Vimont Date: Tue, 4 Sep 2018 08:42:25 +0200 Subject: [PATCH 2/2] Do not handle system-specific values in command.c The common command.c handled process errors from system-specific int values (errno). Rather, expose a new enum process_result to handle error cause in a generic way. --- app/src/command.c | 24 ++++++++---------------- app/src/command.h | 8 +++++++- app/src/sys/unix/command.c | 25 ++++++++++++++++--------- app/src/sys/win/command.c | 8 ++++---- 4 files changed, 35 insertions(+), 30 deletions(-) diff --git a/app/src/command.c b/app/src/command.c index 011a2b85b1..cc11a0de07 100644 --- a/app/src/command.c +++ b/app/src/command.c @@ -1,8 +1,5 @@ #include "command.h" -#ifndef __WINDOWS__ -# include -#endif #include #include #include @@ -21,24 +18,19 @@ static inline const char *get_adb_command() { return adb_command; } -static void show_err_msg(int err) { -#ifdef __WINDOWS__ - (void) err; // unused - LOGE("Failed to execute adb"); -#else +static void show_adb_err_msg(enum process_result err) { switch (err) { - case -1: + case PROCESS_ERROR_GENERIC: LOGE("Failed to execute adb"); break; - case ENOENT: + case PROCESS_ERROR_MISSING_BINARY: LOGE("'adb' command not found (make it accessible from your PATH " "or define its full path in the ADB environment variable)"); break; - default: - LOGE("Failed to execute adb: %s", strerror(err)); + case PROCESS_SUCCESS: + /* do nothing */ break; } -#endif } process_t adb_execute(const char *serial, const char *const adb_cmd[], int len) { @@ -56,9 +48,9 @@ process_t adb_execute(const char *serial, const char *const adb_cmd[], int len) memcpy(&cmd[i], adb_cmd, len * sizeof(const char *)); cmd[len + i] = NULL; - int r = cmd_execute(cmd[0], cmd, &process); - if (r != 0) { - show_err_msg(r); + enum process_result r = cmd_execute(cmd[0], cmd, &process); + if (r != PROCESS_SUCCESS) { + show_adb_err_msg(r); return PROCESS_NONE; } return process; diff --git a/app/src/command.h b/app/src/command.h index dfc0d58a33..3e0fcca699 100644 --- a/app/src/command.h +++ b/app/src/command.h @@ -32,7 +32,13 @@ #endif # define NO_EXIT_CODE -1 -int cmd_execute(const char *path, const char *const argv[], process_t *process); +enum process_result { + PROCESS_SUCCESS, + PROCESS_ERROR_GENERIC, + PROCESS_ERROR_MISSING_BINARY, +}; + +enum process_result cmd_execute(const char *path, const char *const argv[], process_t *process); SDL_bool cmd_terminate(process_t pid); SDL_bool cmd_simple_wait(process_t pid, exit_code_t *exit_code); diff --git a/app/src/sys/unix/command.c b/app/src/sys/unix/command.c index 20218cb9e3..ed9bbeacb6 100644 --- a/app/src/sys/unix/command.c +++ b/app/src/sys/unix/command.c @@ -9,19 +9,20 @@ #include #include "log.h" -int cmd_execute(const char *path, const char *const argv[], pid_t *pid) { +enum process_result cmd_execute(const char *path, const char *const argv[], pid_t *pid) { int fd[2]; - int ret = 0; if (pipe(fd) == -1) { perror("pipe"); - return -1; + return PROCESS_ERROR_GENERIC; } + enum process_result ret = PROCESS_SUCCESS; + *pid = fork(); if (*pid == -1) { perror("fork"); - ret = -1; + ret = PROCESS_ERROR_GENERIC; goto end; } @@ -30,9 +31,9 @@ int cmd_execute(const char *path, const char *const argv[], pid_t *pid) { close(fd[1]); fd[1] = -1; // wait for EOF or receive errno from child - if (read(fd[0], &ret, sizeof(int)) == -1) { + if (read(fd[0], &ret, sizeof(ret)) == -1) { perror("read"); - ret = -1; + ret = PROCESS_ERROR_GENERIC; goto end; } } else if (*pid == 0) { @@ -40,12 +41,18 @@ int cmd_execute(const char *path, const char *const argv[], pid_t *pid) { close(fd[0]); if (fcntl(fd[1], F_SETFD, FD_CLOEXEC) == 0) { execvp(path, (char *const *)argv); + if (errno == ENOENT) { + ret = PROCESS_ERROR_MISSING_BINARY; + } else { + ret = PROCESS_ERROR_GENERIC; + } + perror("exec"); } else { perror("fcntl"); + ret = PROCESS_ERROR_GENERIC; } - // send errno to the parent - ret = errno; - if (write(fd[1], &ret, sizeof(int)) == -1) { + // send ret to the parent + if (write(fd[1], &ret, sizeof(ret)) == -1) { perror("write"); } // close write side before exiting diff --git a/app/src/sys/win/command.c b/app/src/sys/win/command.c index cea928b40d..7a29442ddf 100644 --- a/app/src/sys/win/command.c +++ b/app/src/sys/win/command.c @@ -4,7 +4,7 @@ #include "log.h" #include "str_util.h" -int cmd_execute(const char *path, const char *const argv[], HANDLE *handle) { +enum process_result cmd_execute(const char *path, const char *const argv[], HANDLE *handle) { STARTUPINFO si; PROCESS_INFORMATION pi; memset(&si, 0, sizeof(si)); @@ -19,7 +19,7 @@ int cmd_execute(const char *path, const char *const argv[], HANDLE *handle) { if (ret >= sizeof(cmd)) { LOGE("Command too long (%" PRIsizet " chars)", sizeof(cmd) - 1); *handle = NULL; - return -1; + return PROCESS_ERROR_GENERIC; } #ifdef WINDOWS_NOCONSOLE @@ -29,11 +29,11 @@ int cmd_execute(const char *path, const char *const argv[], HANDLE *handle) { #endif if (!CreateProcess(NULL, cmd, NULL, NULL, FALSE, flags, NULL, NULL, &si, &pi)) { *handle = NULL; - return -1; + return PROCESS_ERROR_GENERIC; } *handle = pi.hProcess; - return 0; + return PROCESS_SUCCESS; } SDL_bool cmd_terminate(HANDLE handle) {