13 #include <sys/types.h> 16 #include <sys/socket.h> 34 strncmp((
const char *)stringval,
"human_readable", strlen(
"human_readable")) == 0);
36 strncmp((
const char *)stringval,
"loaded_config_file_name", strlen(
"loaded_config_file_name")) == 0);
59 if (socket_path == NULL)
63 if (pid_from_atom == NULL) {
65 printf(
"\nRunning version: < 4.2-200\n");
71 printf(
"(Getting version from running i3, press ctrl-c to abort…)");
75 int sockfd = socket(AF_LOCAL, SOCK_STREAM, 0);
77 err(EXIT_FAILURE,
"Could not create socket");
79 struct sockaddr_un addr;
80 memset(&addr, 0,
sizeof(
struct sockaddr_un));
81 addr.sun_family = AF_LOCAL;
82 strncpy(addr.sun_path, socket_path,
sizeof(addr.sun_path) - 1);
83 if (connect(sockfd, (
const struct sockaddr *)&addr,
sizeof(
struct sockaddr_un)) < 0)
84 err(EXIT_FAILURE,
"Could not connect to i3");
88 err(EXIT_FAILURE,
"IPC: write()");
90 uint32_t reply_length;
94 if ((ret =
ipc_recv_message(sockfd, &reply_type, &reply_length, &reply)) != 0) {
96 err(EXIT_FAILURE,
"IPC: read()");
100 if (reply_type != I3_IPC_MESSAGE_TYPE_GET_VERSION)
101 errx(EXIT_FAILURE,
"Got reply type %d, but expected %d (GET_VERSION)", reply_type, I3_IPC_MESSAGE_TYPE_GET_VERSION);
105 yajl_status
state = yajl_parse(handle, (
const unsigned char *)reply, (
int)reply_length);
106 if (state != yajl_status_ok)
107 errx(EXIT_FAILURE,
"Could not parse my own reply. That's weird. reply is %.*s", (
int)reply_length, reply);
120 strftime(mtime,
sizeof(mtime),
"%c", localtime(&(sb.st_mtime)));
122 printf(
" (Last modified: %s, %.f seconds ago)\n", mtime, difftime(now, sb.st_mtime));
127 size_t destpath_size = 1024;
130 char *destpath =
smalloc(destpath_size);
132 sasprintf(&exepath,
"/proc/%d/exe", getpid());
134 while ((linksize = readlink(exepath, destpath, destpath_size)) == (ssize_t)destpath_size) {
135 destpath_size = destpath_size * 2;
136 destpath =
srealloc(destpath, destpath_size);
139 err(EXIT_FAILURE,
"readlink(%s)", exepath);
142 destpath[linksize] =
'\0';
145 printf(
"The i3 binary you just called: %s\n", destpath);
148 sasprintf(&exepath,
"/proc/%s/exe", pid_from_atom);
150 while ((linksize = readlink(exepath, destpath, destpath_size)) == (ssize_t)destpath_size) {
151 destpath_size = destpath_size * 2;
152 destpath =
srealloc(destpath, destpath_size);
155 err(EXIT_FAILURE,
"readlink(%s)", exepath);
158 destpath[linksize] =
'\0';
162 if (strstr(destpath,
"(deleted)") != NULL)
163 printf(
"RUNNING BINARY DIFFERENT FROM BINARY ON DISK!\n");
169 sasprintf(&exepath,
"/proc/%s/cmdline", pid_from_atom);
172 if ((fd = open(exepath, O_RDONLY)) == -1)
173 err(EXIT_FAILURE,
"open(%s)", exepath);
174 if (read(fd, destpath,
sizeof(destpath)) == -1)
175 err(EXIT_FAILURE,
"read(%s)", exepath);
178 printf(
"The i3 binary you are running: %s\n", destpath);
int sasprintf(char **strp, const char *fmt,...)
Safe-wrapper around asprintf which exits if it returns -1 (meaning that there is no more memory avail...
static yajl_callbacks version_callbacks
xcb_connection_t * conn
XCB connection and root screen.
int ipc_send_message(int sockfd, const uint32_t message_size, const uint32_t message_type, const uint8_t *payload)
Formats a message (payload) of the given size and type and sends it to i3 via the given socket file d...
static bool human_readable_key
char * root_atom_contents(const char *atomname, xcb_connection_t *provided_conn, int screen)
Try to get the contents of the given atom (for example I3_SOCKET_PATH) from the X11 root window and r...
static char * loaded_config_file_name
static bool loaded_config_file_name_key
void display_running_version(void)
Connects to i3 to find out the currently running version.
void * smalloc(size_t size)
Safe-wrapper around malloc which exits if malloc returns NULL (meaning that there is no more memory a...
static int version_map_key(void *ctx, const unsigned char *stringval, size_t stringlen)
void * srealloc(void *ptr, size_t size)
Safe-wrapper around realloc which exits if realloc returns NULL (meaning that there is no more memory...
int ipc_recv_message(int sockfd, uint32_t *message_type, uint32_t *reply_length, uint8_t **reply)
Reads a message from the given socket file descriptor and stores its length (reply_length) as well as...
static int version_string(void *ctx, const unsigned char *val, size_t len)
static char * human_readable_version
static xcb_cursor_context_t * ctx