2011-10-02 16:11:30 +01:00
|
|
|
|
/*
|
|
|
|
|
* vim:ts=4:sw=4:expandtab
|
|
|
|
|
*
|
|
|
|
|
* i3 - an improved dynamic tiling window manager
|
2015-04-03 20:17:56 -04:00
|
|
|
|
* © 2009 Michael Stapelberg and contributors (see also: LICENSE)
|
2011-10-02 16:11:30 +01:00
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
#include <limits.h>
|
2013-11-23 11:56:28 +01:00
|
|
|
|
#include <stdlib.h>
|
2013-11-23 12:03:55 +01:00
|
|
|
|
#include <math.h>
|
2011-10-02 16:11:30 +01:00
|
|
|
|
|
|
|
|
|
#include <xcb/xcb.h>
|
|
|
|
|
#include <xcb/xcb_aux.h>
|
|
|
|
|
|
2011-10-23 17:38:21 +01:00
|
|
|
|
#include "libi3.h"
|
|
|
|
|
|
2011-10-02 16:11:30 +01:00
|
|
|
|
/*
|
2012-01-06 23:40:07 +00:00
|
|
|
|
* Try to get the contents of the given atom (for example I3_SOCKET_PATH) from
|
|
|
|
|
* the X11 root window and return NULL if it doesn’t work.
|
2011-10-02 16:11:30 +01:00
|
|
|
|
*
|
2013-11-22 15:48:45 +01:00
|
|
|
|
* If the provided XCB connection is NULL, a new connection will be
|
|
|
|
|
* established.
|
|
|
|
|
*
|
2012-01-06 23:40:07 +00:00
|
|
|
|
* The memory for the contents is dynamically allocated and has to be
|
2011-10-02 16:11:30 +01:00
|
|
|
|
* free()d by the caller.
|
|
|
|
|
*
|
|
|
|
|
*/
|
2013-11-22 15:48:45 +01:00
|
|
|
|
char *root_atom_contents(const char *atomname, xcb_connection_t *provided_conn, int screen) {
|
2011-10-02 16:11:30 +01:00
|
|
|
|
xcb_intern_atom_cookie_t atom_cookie;
|
|
|
|
|
xcb_intern_atom_reply_t *atom_reply;
|
2016-01-08 20:15:34 +01:00
|
|
|
|
char *content = NULL;
|
2013-11-23 12:03:55 +01:00
|
|
|
|
size_t content_max_words = 256;
|
2013-11-22 15:48:45 +01:00
|
|
|
|
xcb_connection_t *conn = provided_conn;
|
2011-10-02 16:11:30 +01:00
|
|
|
|
|
2013-11-22 15:48:45 +01:00
|
|
|
|
if (provided_conn == NULL &&
|
|
|
|
|
((conn = xcb_connect(NULL, &screen)) == NULL ||
|
2016-01-08 20:15:34 +01:00
|
|
|
|
xcb_connection_has_error(conn))) {
|
2011-10-02 16:11:30 +01:00
|
|
|
|
return NULL;
|
2016-01-08 20:15:34 +01:00
|
|
|
|
}
|
2011-10-02 16:11:30 +01:00
|
|
|
|
|
2012-01-06 23:40:07 +00:00
|
|
|
|
atom_cookie = xcb_intern_atom(conn, 0, strlen(atomname), atomname);
|
2011-10-02 16:11:30 +01:00
|
|
|
|
|
|
|
|
|
xcb_screen_t *root_screen = xcb_aux_get_screen(conn, screen);
|
|
|
|
|
xcb_window_t root = root_screen->root;
|
|
|
|
|
|
|
|
|
|
atom_reply = xcb_intern_atom_reply(conn, atom_cookie, NULL);
|
2016-01-08 20:15:34 +01:00
|
|
|
|
if (atom_reply == NULL) {
|
|
|
|
|
goto out_conn;
|
|
|
|
|
}
|
2011-10-02 16:11:30 +01:00
|
|
|
|
|
|
|
|
|
xcb_get_property_cookie_t prop_cookie;
|
|
|
|
|
xcb_get_property_reply_t *prop_reply;
|
|
|
|
|
prop_cookie = xcb_get_property_unchecked(conn, false, root, atom_reply->atom,
|
2013-11-23 12:03:55 +01:00
|
|
|
|
XCB_GET_PROPERTY_TYPE_ANY, 0, content_max_words);
|
2011-10-02 16:11:30 +01:00
|
|
|
|
prop_reply = xcb_get_property_reply(conn, prop_cookie, NULL);
|
2013-11-23 11:56:28 +01:00
|
|
|
|
if (prop_reply == NULL) {
|
2016-01-08 20:15:34 +01:00
|
|
|
|
goto out_atom;
|
2013-11-23 11:56:28 +01:00
|
|
|
|
}
|
2013-11-23 12:03:55 +01:00
|
|
|
|
if (xcb_get_property_value_length(prop_reply) > 0 && prop_reply->bytes_after > 0) {
|
|
|
|
|
/* We received an incomplete value. Ask again but with a properly
|
|
|
|
|
* adjusted size. */
|
|
|
|
|
content_max_words += ceil(prop_reply->bytes_after / 4.0);
|
|
|
|
|
/* Repeat the request, with adjusted size */
|
|
|
|
|
free(prop_reply);
|
|
|
|
|
prop_cookie = xcb_get_property_unchecked(conn, false, root, atom_reply->atom,
|
|
|
|
|
XCB_GET_PROPERTY_TYPE_ANY, 0, content_max_words);
|
|
|
|
|
prop_reply = xcb_get_property_reply(conn, prop_cookie, NULL);
|
|
|
|
|
if (prop_reply == NULL) {
|
2016-01-08 20:15:34 +01:00
|
|
|
|
goto out_atom;
|
2013-11-23 12:03:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2013-11-23 11:56:28 +01:00
|
|
|
|
if (xcb_get_property_value_length(prop_reply) == 0) {
|
2016-01-08 20:15:34 +01:00
|
|
|
|
goto out;
|
2013-11-23 11:56:28 +01:00
|
|
|
|
}
|
2012-08-12 14:10:48 +02:00
|
|
|
|
if (prop_reply->type == XCB_ATOM_CARDINAL) {
|
|
|
|
|
/* We treat a CARDINAL as a >= 32-bit unsigned int. The only CARDINAL
|
|
|
|
|
* we query is I3_PID, which is 32-bit. */
|
2015-08-03 12:50:50 +03:00
|
|
|
|
sasprintf(&content, "%u", *((unsigned int *)xcb_get_property_value(prop_reply)));
|
2012-08-12 14:10:48 +02:00
|
|
|
|
} else {
|
2015-08-03 12:50:50 +03:00
|
|
|
|
sasprintf(&content, "%.*s", xcb_get_property_value_length(prop_reply),
|
|
|
|
|
(char *)xcb_get_property_value(prop_reply));
|
2012-08-12 14:10:48 +02:00
|
|
|
|
}
|
2016-01-08 20:15:34 +01:00
|
|
|
|
|
|
|
|
|
out:
|
|
|
|
|
free(prop_reply);
|
|
|
|
|
out_atom:
|
|
|
|
|
free(atom_reply);
|
|
|
|
|
out_conn:
|
2013-11-22 15:48:45 +01:00
|
|
|
|
if (provided_conn == NULL)
|
|
|
|
|
xcb_disconnect(conn);
|
2012-08-12 14:10:48 +02:00
|
|
|
|
return content;
|
2011-10-02 16:11:30 +01:00
|
|
|
|
}
|