]> git.sur5r.net Git - i3/i3/commitdiff
libi3/root_atom_contents: handle data of arbitrary length
authorLancelot SIX <lancelot@lancleotsix.com>
Sat, 23 Nov 2013 11:03:55 +0000 (12:03 +0100)
committerMichael Stapelberg <michael@stapelberg.de>
Tue, 26 Nov 2013 18:58:35 +0000 (19:58 +0100)
Handle data fetched from xcb_get_property_unchecked with arbitrary
length. This avoids having to rely on PATH_MAX macro where it is not
necessary.

common.mk
libi3/root_atom_contents.c

index 0214abfa52d9e18d956abff02ac851afb5edfc84..e3c92fa1f0846c80a091cd61c4507193fa686cc8 100644 (file)
--- a/common.mk
+++ b/common.mk
@@ -144,7 +144,7 @@ PANGO_LIBS   := $(call ldflags_for_lib, cairo)
 PANGO_LIBS   += $(call ldflags_for_lib, pangocairo)
 
 # libi3
-LIBS = -L$(TOPDIR) -li3
+LIBS = -L$(TOPDIR) -li3 -lm
 
 ## Platform-specific flags
 
index 236f1b994d03aa13cc98ae2d7a2872167de50674..00b74005ba4ef6d8957cc2efb919028218b87b3c 100644 (file)
@@ -10,6 +10,7 @@
 #include <stdbool.h>
 #include <limits.h>
 #include <stdlib.h>
+#include <math.h>
 
 #include <xcb/xcb.h>
 #include <xcb/xcb_aux.h>
@@ -31,6 +32,7 @@ char *root_atom_contents(const char *atomname, xcb_connection_t *provided_conn,
     xcb_intern_atom_cookie_t atom_cookie;
     xcb_intern_atom_reply_t *atom_reply;
     char *content;
+    size_t content_max_words = 256;
     xcb_connection_t *conn = provided_conn;
 
     if (provided_conn == NULL &&
@@ -50,12 +52,26 @@ char *root_atom_contents(const char *atomname, xcb_connection_t *provided_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) {
         free(atom_reply);
         return NULL;
     }
+    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) {
+            free(atom_reply);
+            return NULL;
+        }
+    }
     if (xcb_get_property_value_length(prop_reply) == 0) {
         free(atom_reply);
         free(prop_reply);