]> git.sur5r.net Git - i3/i3/blob - libi3/root_atom_contents.c
00b74005ba4ef6d8957cc2efb919028218b87b3c
[i3/i3] / libi3 / root_atom_contents.c
1 /*
2  * vim:ts=4:sw=4:expandtab
3  *
4  * i3 - an improved dynamic tiling window manager
5  * © 2009-2011 Michael Stapelberg and contributors (see also: LICENSE)
6  *
7  */
8 #include <stdio.h>
9 #include <string.h>
10 #include <stdbool.h>
11 #include <limits.h>
12 #include <stdlib.h>
13 #include <math.h>
14
15 #include <xcb/xcb.h>
16 #include <xcb/xcb_aux.h>
17
18 #include "libi3.h"
19
20 /*
21  * Try to get the contents of the given atom (for example I3_SOCKET_PATH) from
22  * the X11 root window and return NULL if it doesn’t work.
23  *
24  * If the provided XCB connection is NULL, a new connection will be
25  * established.
26  *
27  * The memory for the contents is dynamically allocated and has to be
28  * free()d by the caller.
29  *
30  */
31 char *root_atom_contents(const char *atomname, xcb_connection_t *provided_conn, int screen) {
32     xcb_intern_atom_cookie_t atom_cookie;
33     xcb_intern_atom_reply_t *atom_reply;
34     char *content;
35     size_t content_max_words = 256;
36     xcb_connection_t *conn = provided_conn;
37
38     if (provided_conn == NULL &&
39         ((conn = xcb_connect(NULL, &screen)) == NULL ||
40          xcb_connection_has_error(conn)))
41         return NULL;
42
43     atom_cookie = xcb_intern_atom(conn, 0, strlen(atomname), atomname);
44
45     xcb_screen_t *root_screen = xcb_aux_get_screen(conn, screen);
46     xcb_window_t root = root_screen->root;
47
48     atom_reply = xcb_intern_atom_reply(conn, atom_cookie, NULL);
49     if (atom_reply == NULL)
50         return NULL;
51
52     xcb_get_property_cookie_t prop_cookie;
53     xcb_get_property_reply_t *prop_reply;
54     prop_cookie = xcb_get_property_unchecked(conn, false, root, atom_reply->atom,
55                                              XCB_GET_PROPERTY_TYPE_ANY, 0, content_max_words);
56     prop_reply = xcb_get_property_reply(conn, prop_cookie, NULL);
57     if (prop_reply == NULL) {
58         free(atom_reply);
59         return NULL;
60     }
61     if (xcb_get_property_value_length(prop_reply) > 0 && prop_reply->bytes_after > 0) {
62         /* We received an incomplete value. Ask again but with a properly
63          * adjusted size. */
64         content_max_words += ceil(prop_reply->bytes_after / 4.0);
65         /* Repeat the request, with adjusted size */
66         free(prop_reply);
67         prop_cookie = xcb_get_property_unchecked(conn, false, root, atom_reply->atom,
68                                                  XCB_GET_PROPERTY_TYPE_ANY, 0, content_max_words);
69         prop_reply = xcb_get_property_reply(conn, prop_cookie, NULL);
70         if (prop_reply == NULL) {
71             free(atom_reply);
72             return NULL;
73         }
74     }
75     if (xcb_get_property_value_length(prop_reply) == 0) {
76         free(atom_reply);
77         free(prop_reply);
78         return NULL;
79     }
80     if (prop_reply->type == XCB_ATOM_CARDINAL) {
81         /* We treat a CARDINAL as a >= 32-bit unsigned int. The only CARDINAL
82          * we query is I3_PID, which is 32-bit. */
83         if (asprintf(&content, "%u", *((unsigned int*)xcb_get_property_value(prop_reply))) == -1) {
84             free(atom_reply);
85             free(prop_reply);
86             return NULL;
87         }
88     } else {
89         if (asprintf(&content, "%.*s", xcb_get_property_value_length(prop_reply),
90                      (char*)xcb_get_property_value(prop_reply)) == -1) {
91             free(atom_reply);
92             free(prop_reply);
93             return NULL;
94         }
95     }
96     if (provided_conn == NULL)
97         xcb_disconnect(conn);
98     free(atom_reply);
99     free(prop_reply);
100     return content;
101 }
102