From b3970b36a53ee51ce029fc24f60968e2f46790bc Mon Sep 17 00:00:00 2001 From: Axel Wagner Date: Sun, 26 Dec 2010 19:29:57 +0100 Subject: [PATCH] Bugfix: Recover from closed socket --- i3bar/CHANGELOG | 2 + i3bar/include/ipc.h | 6 +++ i3bar/src/ipc.c | 89 ++++++++++++++++++++++++++++++++------------- i3bar/src/main.c | 4 +- 4 files changed, 74 insertions(+), 27 deletions(-) diff --git a/i3bar/CHANGELOG b/i3bar/CHANGELOG index b8030e43..cbacfb99 100644 --- a/i3bar/CHANGELOG +++ b/i3bar/CHANGELOG @@ -1,3 +1,5 @@ +- Bugfix: Recover from closed socket + v0.6 ===== - Add manpage diff --git a/i3bar/include/ipc.h b/i3bar/include/ipc.h index 0eba64fa..ee273a45 100644 --- a/i3bar/include/ipc.h +++ b/i3bar/include/ipc.h @@ -18,6 +18,12 @@ */ int init_connection(const char *socket_path); +/* + * Destroy the connection to i3. + * + */ +void destroy_connection(); + /* * Sends a Message to i3. * type must be a valid I3_IPC_MESSAGE_TYPE (see i3/ipc.h for further information) diff --git a/i3bar/src/ipc.c b/i3bar/src/ipc.c index 0ddccc25..bd7de75e 100644 --- a/i3bar/src/ipc.c +++ b/i3bar/src/ipc.c @@ -19,30 +19,40 @@ #include "common.h" -ev_io *i3_connection; +ev_io i3_connection; +ev_timer reconn; + +const char *sock_path; typedef void(*handler_t)(char*); /* - * Get a connect to the IPC-interface of i3 and return a filedescriptor + * Retry to connect. * */ -int get_ipc_fd(const char *socket_path) { - int sockfd = socket(AF_LOCAL, SOCK_STREAM, 0); - if (sockfd == -1) { - ELOG("Could not create Socket!\n"); - exit(EXIT_FAILURE); +void retry_connection(struct ev_loop *loop, ev_timer *w, int events) { + static int retries = 8; + if (init_connection(sock_path) == 0) { + if (retries == 0) { + ELOG("Retried 8 times - connection failed!\n"); + exit(EXIT_FAILURE); + } + retries--; + return; } + retries = 8; + ev_timer_stop(loop, w); + subscribe_events(); + reconfig_windows(); +} - struct sockaddr_un addr; - memset(&addr, 0, sizeof(struct sockaddr_un)); - addr.sun_family = AF_LOCAL; - strcpy(addr.sun_path, socket_path); - if (connect(sockfd, (const struct sockaddr*) &addr, sizeof(struct sockaddr_un)) < 0) { - ELOG("Could not connct to i3!\n"); - exit(EXIT_FAILURE); - } - return sockfd; +/* + * Schedule a reconnect + * + */ +void reconnect() { + ev_timer_init(&reconn, retry_connection, 0.25, 0.25); + ev_timer_start(main_loop, &reconn); } /* @@ -144,8 +154,12 @@ void got_data(struct ev_loop *loop, ev_io *watcher, int events) { exit(EXIT_FAILURE); } if (n == 0) { - ELOG("Nothing to read!\n"); - exit(EXIT_FAILURE); + /* EOF received. We try to recover a few times, because most likely + * i3 just restarted */ + ELOG("EOF received, try to recover...\n"); + destroy_connection(); + reconnect(); + return; } rec += n; } @@ -167,8 +181,12 @@ void got_data(struct ev_loop *loop, ev_io *watcher, int events) { * of the message */ char *buffer = malloc(size + 1); if (buffer == NULL) { - ELOG("Could not allocate memory!\n"); - exit(EXIT_FAILURE); + /* EOF received. We try to recover a few times, because most likely + * i3 just restarted */ + ELOG("EOF received, try to recover...\n"); + destroy_connection(); + reconnect(); + return; } rec = 0; @@ -234,7 +252,7 @@ int i3_send_msg(uint32_t type, const char *payload) { uint32_t written = 0; while (to_write > 0) { - int n = write(i3_connection->fd, buffer + written, to_write); + int n = write(i3_connection.fd, buffer + written, to_write); if (n == -1) { ELOG("write() failed!\n"); exit(EXIT_FAILURE); @@ -255,15 +273,36 @@ int i3_send_msg(uint32_t type, const char *payload) { * */ int init_connection(const char *socket_path) { - int sockfd = get_ipc_fd(socket_path); + sock_path = socket_path; + int sockfd = socket(AF_LOCAL, SOCK_STREAM, 0); + if (sockfd == -1) { + ELOG("Could not create Socket!\n"); + exit(EXIT_FAILURE); + } - i3_connection = malloc(sizeof(ev_io)); - ev_io_init(i3_connection, &got_data, sockfd, EV_READ); - ev_io_start(main_loop, i3_connection); + struct sockaddr_un addr; + memset(&addr, 0, sizeof(struct sockaddr_un)); + addr.sun_family = AF_LOCAL; + strcpy(addr.sun_path, sock_path); + if (connect(sockfd, (const struct sockaddr*) &addr, sizeof(struct sockaddr_un)) < 0) { + ELOG("Could not connct to i3!\n"); + reconnect(); + return 0; + } + ev_io_init(&i3_connection, &got_data, sockfd, EV_READ); + ev_io_start(main_loop, &i3_connection); return 1; } +/* + * Destroy the connection to i3. + */ +void destroy_connection() { + close(i3_connection.fd); + ev_io_stop(main_loop, &i3_connection); +} + /* * Subscribe to all the i3-events, we need * diff --git a/i3bar/src/main.c b/i3bar/src/main.c index ec85d324..7f27bdac 100644 --- a/i3bar/src/main.c +++ b/i3bar/src/main.c @@ -215,8 +215,6 @@ int main(int argc, char **argv) { init_outputs(); init_connection(socket_path); - FREE(socket_path); - /* We subscribe to the i3-events we need */ subscribe_events(); @@ -249,6 +247,8 @@ int main(int argc, char **argv) { kill_child(); + FREE(socket_path); + FREE(statusline); clean_xcb();