From 222f0a133cb73a9e8cf747e1f841b07a581f09df Mon Sep 17 00:00:00 2001 From: Uli Schlachter Date: Sun, 27 Mar 2016 11:36:23 +0200 Subject: [PATCH] Keep the cairo surface we use for setting the wallpaper This fixes a race where we would keep giving the old wallpaper surface to Lua when a new one was set, because we didn't get the "wallpaper changed"-event yet. Fix this by just keeping the surface we use for setting the wallpaper around. Signed-off-by: Uli Schlachter --- root.c | 20 ++++++++++++++++++-- 1 file changed, 18 insertions(+), 2 deletions(-) diff --git a/root.c b/root.c index 6b1ab5c5..6682d910 100644 --- a/root.c +++ b/root.c @@ -74,6 +74,7 @@ root_set_wallpaper_pixmap(xcb_connection_t *c, xcb_pixmap_t p) static bool root_set_wallpaper(cairo_pattern_t *pattern) { + lua_State *L = globalconf_get_lua_State(); xcb_connection_t *c = xcb_connect(NULL, NULL); xcb_pixmap_t p = xcb_generate_id(c); /* globalconf.connection should be connected to the same X11 server, so we @@ -107,15 +108,30 @@ root_set_wallpaper(cairo_pattern_t *pattern) cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE); cairo_paint(cr); cairo_destroy(cr); - cairo_surface_finish(surface); - cairo_surface_destroy(surface); + cairo_surface_flush(surface); xcb_aux_sync(globalconf.connection); + /* Change the wallpaper, without sending us a PropertyNotify event */ + xcb_grab_server(globalconf.connection); + xcb_change_window_attributes(globalconf.connection, + globalconf.screen->root, + XCB_CW_EVENT_MASK, + (uint32_t[]) { 0 }); root_set_wallpaper_pixmap(c, p); + xcb_change_window_attributes(globalconf.connection, + globalconf.screen->root, + XCB_CW_EVENT_MASK, + ROOT_WINDOW_EVENT_MASK); + xcb_ungrab_server(globalconf.connection); /* Make sure our pixmap is not destroyed when we disconnect. */ xcb_set_close_down_mode(c, XCB_CLOSE_DOWN_RETAIN_PERMANENT); + /* Tell Lua that the wallpaper changed */ + cairo_surface_destroy(globalconf.wallpaper); + globalconf.wallpaper = surface; + signal_object_emit(L, &global_signals, "wallpaper_changed", 0); + result = true; disconnect: xcb_flush(c);