X-Git-Url: https://git.sur5r.net/?a=blobdiff_plain;ds=sidebyside;f=libi3%2Froot_atom_contents.c;h=d6394d4bf2f7631fef52cb89454f32ca3a705664;hb=297abe0eddbbd61ae39e5757441c98ad7d34ccb2;hp=927cc5f8cf1945d0f2baab4531f8c5e8196e9bff;hpb=d4523de6c8f7d930d52ce69d94ff186a29c5e803;p=i3%2Fi3 diff --git a/libi3/root_atom_contents.c b/libi3/root_atom_contents.c index 927cc5f8..d6394d4b 100644 --- a/libi3/root_atom_contents.c +++ b/libi3/root_atom_contents.c @@ -2,37 +2,44 @@ * vim:ts=4:sw=4:expandtab * * i3 - an improved dynamic tiling window manager - * © 2009-2011 Michael Stapelberg and contributors (see also: LICENSE) + * © 2009 Michael Stapelberg and contributors (see also: LICENSE) * */ +#include "libi3.h" + #include #include #include #include +#include +#include #include #include -#include "libi3.h" - /* * 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. * + * If the provided XCB connection is NULL, a new connection will be + * established. + * * The memory for the contents is dynamically allocated and has to be * free()d by the caller. * */ -char *root_atom_contents(const char *atomname) { - xcb_connection_t *conn; +char *root_atom_contents(const char *atomname, xcb_connection_t *provided_conn, int screen) { xcb_intern_atom_cookie_t atom_cookie; xcb_intern_atom_reply_t *atom_reply; - int screen; - char *socket_path; + char *content = NULL; + size_t content_max_words = 256; + xcb_connection_t *conn = provided_conn; - if ((conn = xcb_connect(NULL, &screen)) == NULL || - xcb_connection_has_error(conn)) + if (provided_conn == NULL && + ((conn = xcb_connect(NULL, &screen)) == NULL || + xcb_connection_has_error(conn))) { return NULL; + } atom_cookie = xcb_intern_atom(conn, 0, strlen(atomname), atomname); @@ -40,20 +47,49 @@ char *root_atom_contents(const char *atomname) { xcb_window_t root = root_screen->root; atom_reply = xcb_intern_atom_reply(conn, atom_cookie, NULL); - if (atom_reply == NULL) - return NULL; + if (atom_reply == NULL) { + goto out_conn; + } 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, - XCB_GET_PROPERTY_TYPE_ANY, 0, PATH_MAX); + XCB_GET_PROPERTY_TYPE_ANY, 0, content_max_words); prop_reply = xcb_get_property_reply(conn, prop_cookie, NULL); - if (prop_reply == NULL || xcb_get_property_value_length(prop_reply) == 0) - return NULL; - if (asprintf(&socket_path, "%.*s", xcb_get_property_value_length(prop_reply), - (char*)xcb_get_property_value(prop_reply)) == -1) - return NULL; - xcb_disconnect(conn); - return socket_path; -} + if (prop_reply == NULL) { + goto out_atom; + } + 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) { + goto out_atom; + } + } + if (xcb_get_property_value_length(prop_reply) == 0) { + goto out; + } + 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. */ + sasprintf(&content, "%u", *((unsigned int *)xcb_get_property_value(prop_reply))); + } else { + sasprintf(&content, "%.*s", xcb_get_property_value_length(prop_reply), + (char *)xcb_get_property_value(prop_reply)); + } +out: + free(prop_reply); +out_atom: + free(atom_reply); +out_conn: + if (provided_conn == NULL) + xcb_disconnect(conn); + return content; +}